Add stepping and breakpoints to debugger
This commit is contained in:
parent
38247f03c8
commit
7d9fb457ff
@ -134,6 +134,16 @@ class ExecutionController<RS> {
|
|||||||
return this._result;
|
return this._result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run a single step of execution
|
||||||
|
* @returns Result of execution
|
||||||
|
*/
|
||||||
|
executeStep(): StepExecutionResult<RS> {
|
||||||
|
this._result = this._engine.executeStep();
|
||||||
|
this._result.signal = "paused";
|
||||||
|
return this._result;
|
||||||
|
}
|
||||||
|
|
||||||
/** Asynchronously sleep for a period of time */
|
/** Asynchronously sleep for a period of time */
|
||||||
private async sleep(millis: number) {
|
private async sleep(millis: number) {
|
||||||
return new Promise<void>((resolve) => setTimeout(resolve, millis));
|
return new Promise<void>((resolve) => setTimeout(resolve, millis));
|
||||||
|
@ -22,6 +22,10 @@ export type WorkerRequestData =
|
|||||||
type: "Execute";
|
type: "Execute";
|
||||||
params: { interval?: number };
|
params: { interval?: number };
|
||||||
}
|
}
|
||||||
|
| {
|
||||||
|
type: "ExecuteStep";
|
||||||
|
params?: null;
|
||||||
|
}
|
||||||
| {
|
| {
|
||||||
type: "Pause";
|
type: "Pause";
|
||||||
params?: null;
|
params?: null;
|
||||||
|
@ -64,7 +64,6 @@ const updateBreakpoints = (points: number[]) => {
|
|||||||
* and return result of execution.
|
* and return result of execution.
|
||||||
*/
|
*/
|
||||||
const execute = (interval?: number) => {
|
const execute = (interval?: number) => {
|
||||||
console.info(`Executing at interval ${interval}`);
|
|
||||||
_controller!.executeAll({
|
_controller!.executeAll({
|
||||||
interval,
|
interval,
|
||||||
onResult: (res) => postMessage(resultMessage(res)),
|
onResult: (res) => postMessage(resultMessage(res)),
|
||||||
@ -79,12 +78,21 @@ const pauseExecution = async () => {
|
|||||||
postMessage(ackMessage("pause"));
|
postMessage(ackMessage("pause"));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run a single execution step
|
||||||
|
*/
|
||||||
|
const executeStep = () => {
|
||||||
|
const result = _controller!.executeStep();
|
||||||
|
postMessage(resultMessage(result));
|
||||||
|
};
|
||||||
|
|
||||||
addEventListener("message", async (ev: MessageEvent<WorkerRequestData>) => {
|
addEventListener("message", async (ev: MessageEvent<WorkerRequestData>) => {
|
||||||
if (ev.data.type === "Init") return initController();
|
if (ev.data.type === "Init") return initController();
|
||||||
if (ev.data.type === "Reset") return resetController();
|
if (ev.data.type === "Reset") return resetController();
|
||||||
if (ev.data.type === "Prepare") return prepare(ev.data.params);
|
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 === "Execute") return execute(ev.data.params.interval);
|
||||||
if (ev.data.type === "Pause") return await pauseExecution();
|
if (ev.data.type === "Pause") return await pauseExecution();
|
||||||
|
if (ev.data.type === "ExecuteStep") return executeStep();
|
||||||
if (ev.data.type === "UpdateBreakpoints")
|
if (ev.data.type === "UpdateBreakpoints")
|
||||||
return updateBreakpoints(ev.data.params.points);
|
return updateBreakpoints(ev.data.params.points);
|
||||||
throw new Error("Invalid worker message type");
|
throw new Error("Invalid worker message type");
|
||||||
|
@ -57,6 +57,21 @@ export const Mainframe = () => {
|
|||||||
await execController.pauseExecution();
|
await execController.pauseExecution();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** Run a single step of execution */
|
||||||
|
const executeStep = async () => {
|
||||||
|
// Check if controller is paused
|
||||||
|
if (execController.state !== "paused") {
|
||||||
|
console.error("Controller not paused");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run and update execution states
|
||||||
|
const result = await execController.executeStep();
|
||||||
|
setRendererState(result.rendererState);
|
||||||
|
setCodeHighlights(result.nextStepLocation || undefined);
|
||||||
|
setOutput((o) => (o || "") + (result.output || ""));
|
||||||
|
};
|
||||||
|
|
||||||
/** Resume the currently paused execution */
|
/** Resume the currently paused execution */
|
||||||
const resumeExecution = async () => {
|
const resumeExecution = async () => {
|
||||||
// Check if controller is indeed paused
|
// Check if controller is indeed paused
|
||||||
@ -125,6 +140,7 @@ export const Mainframe = () => {
|
|||||||
onRun={runProgram}
|
onRun={runProgram}
|
||||||
onPause={pauseExecution}
|
onPause={pauseExecution}
|
||||||
onResume={resumeExecution}
|
onResume={resumeExecution}
|
||||||
|
onStep={executeStep}
|
||||||
onStop={stopExecution}
|
onStop={stopExecution}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
@ -24,6 +24,7 @@ const DebugControls = (props: {
|
|||||||
paused: boolean;
|
paused: boolean;
|
||||||
onPause: () => void;
|
onPause: () => void;
|
||||||
onResume: () => void;
|
onResume: () => void;
|
||||||
|
onStep: () => void;
|
||||||
onStop: () => void;
|
onStop: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
return (
|
return (
|
||||||
@ -37,6 +38,7 @@ const DebugControls = (props: {
|
|||||||
<Button
|
<Button
|
||||||
small
|
small
|
||||||
title="Step"
|
title="Step"
|
||||||
|
onClick={props.onStep}
|
||||||
disabled={!props.paused}
|
disabled={!props.paused}
|
||||||
icon={<Icon icon="step-forward" intent="warning" />}
|
icon={<Icon icon="step-forward" intent="warning" />}
|
||||||
/>
|
/>
|
||||||
@ -55,6 +57,7 @@ type Props = {
|
|||||||
onRun: () => void;
|
onRun: () => void;
|
||||||
onPause: () => void;
|
onPause: () => void;
|
||||||
onResume: () => void;
|
onResume: () => void;
|
||||||
|
onStep: () => void;
|
||||||
onStop: () => void;
|
onStop: () => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -68,6 +71,7 @@ export const ExecutionControls = (props: Props) => {
|
|||||||
paused={props.state === "paused"}
|
paused={props.state === "paused"}
|
||||||
onPause={props.onPause}
|
onPause={props.onPause}
|
||||||
onResume={props.onResume}
|
onResume={props.onResume}
|
||||||
|
onStep={props.onStep}
|
||||||
onStop={props.onStop}
|
onStop={props.onStop}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
@ -140,6 +140,20 @@ export const useExecController = <RS>() => {
|
|||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run a single step of execution
|
||||||
|
* @return Execution result
|
||||||
|
*/
|
||||||
|
const executeStep = React.useCallback(async () => {
|
||||||
|
const res = await requestWorker(
|
||||||
|
{ type: "ExecuteStep" },
|
||||||
|
(res) => res.type !== "result"
|
||||||
|
);
|
||||||
|
if (res.type !== "result") throw new Error("Something unexpected happened");
|
||||||
|
if (!res.data.nextStepLocation) setWorkerState("done");
|
||||||
|
return res.data;
|
||||||
|
}, []);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Execute the code loaded into the engine
|
* Execute the code loaded into the engine
|
||||||
* @param onResult Callback used when an execution result is received
|
* @param onResult Callback used when an execution result is received
|
||||||
@ -172,6 +186,7 @@ export const useExecController = <RS>() => {
|
|||||||
prepare,
|
prepare,
|
||||||
pauseExecution,
|
pauseExecution,
|
||||||
execute,
|
execute,
|
||||||
|
executeStep,
|
||||||
updateBreakpoints,
|
updateBreakpoints,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user