[turbolizer] TurboFan nodes history improvements
Added: - history's circles titles - history's records titles - ability to move to node from history view - new hotkey for turboshaft layout Bug: v8:7327 Change-Id: I7ecfdbef2c1bf9534c76f8ac253e846beeea8cb3 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3779909 Reviewed-by: Tobias Tebbi <tebbi@chromium.org> Commit-Queue: Danylo Boiko <danielboyko02@gmail.com> Cr-Commit-Position: refs/heads/main@{#82089}
This commit is contained in:
parent
07e7da140a
commit
614dbbff2f
@ -787,3 +787,8 @@ svg.history-svg-container .history-content-scroll {
|
|||||||
svg.history-svg-container .history-item tspan {
|
svg.history-svg-container .history-item tspan {
|
||||||
font-size: var(--history-item-tspan-font-size);
|
font-size: var(--history-item-tspan-font-size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
svg.history-svg-container .history-item-record tspan:hover {
|
||||||
|
fill: #606060;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
@ -89,9 +89,13 @@
|
|||||||
<td>Copy hovered node's info</td>
|
<td>Copy hovered node's info</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>u</td>
|
<td>y</td>
|
||||||
<td>Collapse unused blocks (blocks that don't have direct inputs and outputs of a hovered node)</td>
|
<td>Collapse unused blocks (blocks that don't have direct inputs and outputs of a hovered node)</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>u</td>
|
||||||
|
<td>Collapse unused blocks (blocks that don't have direct inputs and outputs of selected nodes)</td>
|
||||||
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -24,7 +24,7 @@ export const SOURCE_PANE_DEFAULT_PERCENT = 1 / 4;
|
|||||||
export const DISASSEMBLY_PANE_DEFAULT_PERCENT = 3 / 4;
|
export const DISASSEMBLY_PANE_DEFAULT_PERCENT = 3 / 4;
|
||||||
export const RANGES_PANE_HEIGHT_DEFAULT_PERCENT = 3 / 4;
|
export const RANGES_PANE_HEIGHT_DEFAULT_PERCENT = 3 / 4;
|
||||||
export const RANGES_PANE_WIDTH_DEFAULT_PERCENT = 1 / 2;
|
export const RANGES_PANE_WIDTH_DEFAULT_PERCENT = 1 / 2;
|
||||||
export const HISTORY_DEFAULT_HEIGHT_PERCENT = 1 / 5;
|
export const HISTORY_DEFAULT_HEIGHT_PERCENT = 1 / 3.5;
|
||||||
export const HISTORY_CONTENT_INDENT = 8;
|
export const HISTORY_CONTENT_INDENT = 8;
|
||||||
export const HISTORY_SCROLLBAR_WIDTH = 6;
|
export const HISTORY_SCROLLBAR_WIDTH = 6;
|
||||||
export const CLOSE_BUTTON_RADIUS = 25;
|
export const CLOSE_BUTTON_RADIUS = 25;
|
||||||
|
@ -93,6 +93,13 @@ export class GraphMultiView extends View {
|
|||||||
this.displayPhase(this.sourceResolver.getPhase(initialPhaseIndex));
|
this.displayPhase(this.sourceResolver.getPhase(initialPhaseIndex));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public displayPhaseByName(phaseName: string, selection?: SelectionStorage): void {
|
||||||
|
this.currentPhaseView.hide();
|
||||||
|
const phaseId = this.sourceResolver.getPhaseIdByName(phaseName);
|
||||||
|
this.selectMenu.selectedIndex = phaseId;
|
||||||
|
this.displayPhase(this.sourceResolver.getPhase(phaseId), selection);
|
||||||
|
}
|
||||||
|
|
||||||
public onresize(): void {
|
public onresize(): void {
|
||||||
this.currentPhaseView?.onresize();
|
this.currentPhaseView?.onresize();
|
||||||
}
|
}
|
||||||
@ -116,12 +123,6 @@ export class GraphMultiView extends View {
|
|||||||
this.currentPhaseView = view;
|
this.currentPhaseView = view;
|
||||||
}
|
}
|
||||||
|
|
||||||
private displayPhaseByName(phaseName: string, selection?: SelectionStorage): void {
|
|
||||||
const phaseId = this.sourceResolver.getPhaseIdByName(phaseName);
|
|
||||||
this.selectMenu.selectedIndex = phaseId;
|
|
||||||
this.displayPhase(this.sourceResolver.getPhase(phaseId), selection);
|
|
||||||
}
|
|
||||||
|
|
||||||
private displayNextGraphPhase(): void {
|
private displayNextGraphPhase(): void {
|
||||||
let nextPhaseIndex = this.selectMenu.selectedIndex + 1;
|
let nextPhaseIndex = this.selectMenu.selectedIndex + 1;
|
||||||
while (nextPhaseIndex < this.sourceResolver.phases.length) {
|
while (nextPhaseIndex < this.sourceResolver.phases.length) {
|
||||||
|
@ -86,11 +86,11 @@ export class SelectionBroker {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO (danylo boiko) Add instructionOffsets type
|
public broadcastInstructionSelect(from, instructionOffsets: Array<number>, selected: boolean):
|
||||||
public broadcastInstructionSelect(from, instructionOffsets, selected: boolean): void {
|
void {
|
||||||
// Select the lines from the disassembly (right panel)
|
// Select the lines from the disassembly (right panel)
|
||||||
for (const handler of this.instructionHandlers) {
|
for (const handler of this.instructionHandlers) {
|
||||||
if (handler != from) handler.brokeredInstructionSelect(instructionOffsets, selected);
|
if (handler != from) handler.brokeredInstructionSelect([instructionOffsets], selected);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Select the lines from the source panel (left panel)
|
// Select the lines from the source panel (left panel)
|
||||||
|
@ -31,7 +31,8 @@ export interface BlockSelectionHandler {
|
|||||||
export interface InstructionSelectionHandler {
|
export interface InstructionSelectionHandler {
|
||||||
select(instructionIds: Array<string>, selected: boolean): void;
|
select(instructionIds: Array<string>, selected: boolean): void;
|
||||||
clear(): void;
|
clear(): void;
|
||||||
brokeredInstructionSelect(instructionsOffsets: Array<[number, number]>, selected: boolean): void;
|
brokeredInstructionSelect(instructionsOffsets: Array<[number, number]> | Array<Array<number>>,
|
||||||
|
selected: boolean): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SourcePositionSelectionHandler {
|
export interface SourcePositionSelectionHandler {
|
||||||
@ -42,7 +43,7 @@ export interface SourcePositionSelectionHandler {
|
|||||||
|
|
||||||
export interface RegisterAllocationSelectionHandler {
|
export interface RegisterAllocationSelectionHandler {
|
||||||
// These are called instructionIds since the class of the divs is "instruction-id"
|
// These are called instructionIds since the class of the divs is "instruction-id"
|
||||||
select(instructionIds: Array<string>, selected: boolean): void;
|
select(instructionIds: Array<number>, selected: boolean): void;
|
||||||
clear(): void;
|
clear(): void;
|
||||||
brokeredRegisterAllocationSelect(instructionsOffsets: Array<[number, number]>, selected: boolean):
|
brokeredRegisterAllocationSelect(instructionsOffsets: Array<[number, number]>, selected: boolean):
|
||||||
void;
|
void;
|
||||||
|
@ -97,7 +97,8 @@ window.onload = function () {
|
|||||||
multiview = new GraphMultiView(C.INTERMEDIATE_PANE_ID, selectionBroker, sourceResolver);
|
multiview = new GraphMultiView(C.INTERMEDIATE_PANE_ID, selectionBroker, sourceResolver);
|
||||||
multiview.show();
|
multiview.show();
|
||||||
|
|
||||||
historyView = new HistoryView(C.HISTORY_ID, selectionBroker, sourceResolver);
|
historyView = new HistoryView(C.HISTORY_ID, selectionBroker, sourceResolver,
|
||||||
|
multiview.displayPhaseByName.bind(multiview));
|
||||||
historyView.show();
|
historyView.show();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (window.confirm("Error: Exception during load of TurboFan JSON file:\n" +
|
if (window.confirm("Error: Exception during load of TurboFan JSON file:\n" +
|
||||||
|
@ -196,8 +196,8 @@ export class DisassemblyView extends TextView {
|
|||||||
view.updateSelection();
|
view.updateSelection();
|
||||||
broker.broadcastClear(this);
|
broker.broadcastClear(this);
|
||||||
},
|
},
|
||||||
brokeredInstructionSelect: function (instructionsOffsets: Array<[number, number]>,
|
brokeredInstructionSelect: function (instructionsOffsets: Array<[number, number]> |
|
||||||
selected: boolean) {
|
Array<Array<number>>, selected: boolean) {
|
||||||
const firstSelect = view.offsetSelection.isEmpty();
|
const firstSelect = view.offsetSelection.isEmpty();
|
||||||
for (const instructionOffset of instructionsOffsets) {
|
for (const instructionOffset of instructionsOffsets) {
|
||||||
const keyPcOffsets = view.sourceResolver.instructionsPhase
|
const keyPcOffsets = view.sourceResolver.instructionsPhase
|
||||||
|
@ -28,7 +28,8 @@ export class GraphView extends MovableView<Graph> {
|
|||||||
drag: d3.DragBehavior<any, GraphNode, GraphNode>;
|
drag: d3.DragBehavior<any, GraphNode, GraphNode>;
|
||||||
|
|
||||||
constructor(idOrContainer: string | HTMLElement, broker: SelectionBroker,
|
constructor(idOrContainer: string | HTMLElement, broker: SelectionBroker,
|
||||||
showPhaseByName: (name: string) => void, toolbox: HTMLElement) {
|
showPhaseByName: (name: string, selection: SelectionStorage) => void,
|
||||||
|
toolbox: HTMLElement) {
|
||||||
super(idOrContainer, broker, showPhaseByName, toolbox);
|
super(idOrContainer, broker, showPhaseByName, toolbox);
|
||||||
|
|
||||||
this.state.selection = new SelectionMap(node => node.identifier(),
|
this.state.selection = new SelectionMap(node => node.identifier(),
|
||||||
@ -79,6 +80,7 @@ export class GraphView extends MovableView<Graph> {
|
|||||||
|
|
||||||
if (selectedNodes?.length > 0) {
|
if (selectedNodes?.length > 0) {
|
||||||
this.connectVisibleSelectedElements(this.state.selection);
|
this.connectVisibleSelectedElements(this.state.selection);
|
||||||
|
this.updateGraphVisibility();
|
||||||
this.viewSelection();
|
this.viewSelection();
|
||||||
} else {
|
} else {
|
||||||
if (this.state.cacheLayout && data.transform) {
|
if (this.state.cacheLayout && data.transform) {
|
||||||
@ -367,6 +369,10 @@ export class GraphView extends MovableView<Graph> {
|
|||||||
return new SelectionStorage();
|
return new SelectionStorage();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (const node of rememberedSelection.adaptedNodes) {
|
||||||
|
this.graph.makeNodeVisible(node);
|
||||||
|
}
|
||||||
|
|
||||||
for (const [key, node] of rememberedSelection.nodes.entries()) {
|
for (const [key, node] of rememberedSelection.nodes.entries()) {
|
||||||
// Adding survived nodes (with the same id)
|
// Adding survived nodes (with the same id)
|
||||||
const survivedNode = this.graph.nodeMap[key];
|
const survivedNode = this.graph.nodeMap[key];
|
||||||
@ -711,9 +717,9 @@ export class GraphView extends MovableView<Graph> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private selectOrigins(): void {
|
private selectOrigins(): void {
|
||||||
|
const selection = new SelectionStorage();
|
||||||
const origins = new Array<GraphNode>();
|
const origins = new Array<GraphNode>();
|
||||||
let phase = this.phaseName;
|
let phase = this.phaseName;
|
||||||
const selection = new Set<string>();
|
|
||||||
for (const node of this.state.selection) {
|
for (const node of this.state.selection) {
|
||||||
const origin = node.nodeLabel.origin;
|
const origin = node.nodeLabel.origin;
|
||||||
if (origin && origin instanceof NodeOrigin) {
|
if (origin && origin instanceof NodeOrigin) {
|
||||||
@ -722,14 +728,13 @@ export class GraphView extends MovableView<Graph> {
|
|||||||
if (phase === this.phaseName && node) {
|
if (phase === this.phaseName && node) {
|
||||||
origins.push(node);
|
origins.push(node);
|
||||||
} else {
|
} else {
|
||||||
selection.add(origin.identifier());
|
selection.adaptNode(origin.identifier());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Only go through phase reselection if we actually need
|
// Only go through phase reselection if we actually need
|
||||||
// to display another phase.
|
// to display another phase.
|
||||||
if (selection.size > 0 && phase !== this.phaseName) {
|
if (selection.isAdapted() && phase !== this.phaseName) {
|
||||||
this.hide();
|
|
||||||
this.showPhaseByName(phase, selection);
|
this.showPhaseByName(phase, selection);
|
||||||
} else if (origins.length > 0) {
|
} else if (origins.length > 0) {
|
||||||
this.nodeSelectionHandler.clear();
|
this.nodeSelectionHandler.clear();
|
||||||
|
@ -12,6 +12,7 @@ import { GraphNode } from "../phases/graph-phase/graph-node";
|
|||||||
import { HistoryHandler } from "../selection/selection-handler";
|
import { HistoryHandler } from "../selection/selection-handler";
|
||||||
import { GraphPhase } from "../phases/graph-phase/graph-phase";
|
import { GraphPhase } from "../phases/graph-phase/graph-phase";
|
||||||
import { NodeOrigin } from "../origin";
|
import { NodeOrigin } from "../origin";
|
||||||
|
import { SelectionStorage } from "../selection/selection-storage";
|
||||||
|
|
||||||
export class HistoryView extends View {
|
export class HistoryView extends View {
|
||||||
node: GraphNode;
|
node: GraphNode;
|
||||||
@ -27,11 +28,14 @@ export class HistoryView extends View {
|
|||||||
y: number;
|
y: number;
|
||||||
maxNodeWidth: number;
|
maxNodeWidth: number;
|
||||||
maxPhaseNameWidth: number;
|
maxPhaseNameWidth: number;
|
||||||
|
showPhaseByName: (name: string, selection: SelectionStorage) => void;
|
||||||
|
|
||||||
constructor(id: string, broker: SelectionBroker, sourceResolver: SourceResolver) {
|
constructor(id: string, broker: SelectionBroker, sourceResolver: SourceResolver,
|
||||||
|
showPhaseByName: (name: string, selection: SelectionStorage) => void) {
|
||||||
super(id);
|
super(id);
|
||||||
this.broker = broker;
|
this.broker = broker;
|
||||||
this.sourceResolver = sourceResolver;
|
this.sourceResolver = sourceResolver;
|
||||||
|
this.showPhaseByName = showPhaseByName;
|
||||||
this.historyHandler = this.initializeNodeSelectionHandler();
|
this.historyHandler = this.initializeNodeSelectionHandler();
|
||||||
this.broker.addHistoryHandler(this.historyHandler);
|
this.broker.addHistoryHandler(this.historyHandler);
|
||||||
this.phaseIdToHistory = new Map<number, PhaseHistory>();
|
this.phaseIdToHistory = new Map<number, PhaseHistory>();
|
||||||
@ -60,8 +64,11 @@ export class HistoryView extends View {
|
|||||||
.style("visibility", "hidden");
|
.style("visibility", "hidden");
|
||||||
|
|
||||||
const dragHandler = d3.drag().on("drag", () => {
|
const dragHandler = d3.drag().on("drag", () => {
|
||||||
this.x += d3.event.dx;
|
const rect = document.body.getBoundingClientRect();
|
||||||
this.y += d3.event.dy;
|
const x = this.x + d3.event.dx;
|
||||||
|
this.x = d3.event.dx > 0 ? Math.min(x, rect.width - this.getWidth()) : Math.max(x, 0);
|
||||||
|
const y = this.y + d3.event.dy;
|
||||||
|
this.y = d3.event.dy > 0 ? Math.min(y, rect.height - this.getHeight()) : Math.max(y, 0);
|
||||||
this.svg.attr("transform", _ => `translate(${this.x},${this.y})`);
|
this.svg.attr("transform", _ => `translate(${this.x},${this.y})`);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -125,7 +132,9 @@ export class HistoryView extends View {
|
|||||||
let recordY = 0;
|
let recordY = 0;
|
||||||
|
|
||||||
for (const [phaseId, phaseHistory] of this.phaseIdToHistory.entries()) {
|
for (const [phaseId, phaseHistory] of this.phaseIdToHistory.entries()) {
|
||||||
|
if (!phaseHistory.hasChanges()) continue;
|
||||||
const phaseName = this.sourceResolver.getPhaseNameById(phaseId);
|
const phaseName = this.sourceResolver.getPhaseNameById(phaseId);
|
||||||
|
|
||||||
this.historyList
|
this.historyList
|
||||||
.append("text")
|
.append("text")
|
||||||
.classed("history-item", true)
|
.classed("history-item", true)
|
||||||
@ -164,16 +173,25 @@ export class HistoryView extends View {
|
|||||||
.classed("history-item", true)
|
.classed("history-item", true)
|
||||||
.attr("r", this.labelBox.height / 3.5)
|
.attr("r", this.labelBox.height / 3.5)
|
||||||
.attr("cx", this.labelBox.height / 3)
|
.attr("cx", this.labelBox.height / 3)
|
||||||
.attr("cy", this.labelBox.height / 3 + recordY)
|
.attr("cy", this.labelBox.height / 2.5 + recordY)
|
||||||
.attr("fill", `url(#${circleId})`);
|
.attr("fill", `url(#${circleId})`)
|
||||||
|
.append("title")
|
||||||
|
.text(`[${record.toString()}]`);
|
||||||
|
|
||||||
this.historyList
|
this.historyList
|
||||||
.append("text")
|
.append("text")
|
||||||
.classed("history-item", true)
|
.classed("history-item history-item-record", true)
|
||||||
.attr("dy", recordY)
|
.attr("dy", recordY)
|
||||||
.attr("dx", this.labelBox.height * 0.75)
|
.attr("dx", this.labelBox.height * 0.75)
|
||||||
.append("tspan")
|
.append("tspan")
|
||||||
.text(record.node.displayLabel);
|
.text(record.node.displayLabel)
|
||||||
|
.on("click", () => {
|
||||||
|
const selectionStorage = new SelectionStorage();
|
||||||
|
selectionStorage.adaptNode(record.node.identifier());
|
||||||
|
this.showPhaseByName(phaseName, selectionStorage);
|
||||||
|
})
|
||||||
|
.append("title")
|
||||||
|
.text(record.node.getTitle());
|
||||||
recordY += this.labelBox.height;
|
recordY += this.labelBox.height;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -198,13 +216,6 @@ export class HistoryView extends View {
|
|||||||
content.node().appendChild(d3.select(this).node() as HTMLElement);
|
content.node().appendChild(d3.select(this).node() as HTMLElement);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.historyList
|
|
||||||
.append("rect")
|
|
||||||
.attr("width", historyArea.width)
|
|
||||||
.attr("height", historyArea.height)
|
|
||||||
.attr("transform", `translate(${historyArea.x},${historyArea.y})`)
|
|
||||||
.attr("opacity", 0);
|
|
||||||
|
|
||||||
this.historyList
|
this.historyList
|
||||||
.append("clipPath")
|
.append("clipPath")
|
||||||
.attr("id", "history-clip-path")
|
.attr("id", "history-clip-path")
|
||||||
@ -240,7 +251,7 @@ export class HistoryView extends View {
|
|||||||
if (!isNaN(scrollBarPosition)) scrollBar.attr("y", scrollBarPosition);
|
if (!isNaN(scrollBarPosition)) scrollBar.attr("y", scrollBarPosition);
|
||||||
};
|
};
|
||||||
|
|
||||||
this.historyList.on("wheel", () => {
|
this.svg.on("wheel", () => {
|
||||||
updateScrollPosition(d3.event.deltaY);
|
updateScrollPosition(d3.event.deltaY);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -267,6 +278,7 @@ export class HistoryView extends View {
|
|||||||
const uniqueAncestors = new Set<string>();
|
const uniqueAncestors = new Set<string>();
|
||||||
const coefficient = this.getCoefficient("history-item-tspan-font-size");
|
const coefficient = this.getCoefficient("history-item-tspan-font-size");
|
||||||
let prevNode = null;
|
let prevNode = null;
|
||||||
|
let first = true;
|
||||||
for (let i = 0; i < this.sourceResolver.phases.length; i++) {
|
for (let i = 0; i < this.sourceResolver.phases.length; i++) {
|
||||||
const phase = this.sourceResolver.getPhase(i);
|
const phase = this.sourceResolver.getPhase(i);
|
||||||
if (!(phase instanceof GraphPhase)) continue;
|
if (!(phase instanceof GraphPhase)) continue;
|
||||||
@ -305,6 +317,11 @@ export class HistoryView extends View {
|
|||||||
this.addToHistory(i, node, HistoryChange.InplaceUpdated);
|
this.addToHistory(i, node, HistoryChange.InplaceUpdated);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (first) {
|
||||||
|
this.addToHistory(i, node, HistoryChange.Emerged);
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
|
|
||||||
this.addToHistory(i, node, HistoryChange.Current);
|
this.addToHistory(i, node, HistoryChange.Current);
|
||||||
prevNode = node;
|
prevNode = node;
|
||||||
}
|
}
|
||||||
@ -395,18 +412,19 @@ export class HistoryView extends View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private getHeight(): number {
|
private getHeight(): number {
|
||||||
const clientRect = document.body.getBoundingClientRect();
|
return window.screen.availHeight * C.HISTORY_DEFAULT_HEIGHT_PERCENT;
|
||||||
return clientRect.height * C.HISTORY_DEFAULT_HEIGHT_PERCENT;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private getHistoryChangeColor(historyChange: HistoryChange): string {
|
private getHistoryChangeColor(historyChange: HistoryChange): string {
|
||||||
switch (historyChange) {
|
switch (historyChange) {
|
||||||
case HistoryChange.Current:
|
case HistoryChange.Current:
|
||||||
return "rgb(255, 167, 0)";
|
return "rgb(255, 167, 0)";
|
||||||
|
case HistoryChange.Emerged:
|
||||||
|
return "rgb(160, 83, 236)";
|
||||||
case HistoryChange.Lowered:
|
case HistoryChange.Lowered:
|
||||||
return "rgb(0, 255, 0)";
|
return "rgb(0, 255, 0)";
|
||||||
case HistoryChange.InplaceUpdated:
|
case HistoryChange.InplaceUpdated:
|
||||||
return "rgb(0, 0, 255)";
|
return "rgb(57, 57, 208)";
|
||||||
case HistoryChange.Removed:
|
case HistoryChange.Removed:
|
||||||
return "rgb(255, 0, 0)";
|
return "rgb(255, 0, 0)";
|
||||||
case HistoryChange.Survived:
|
case HistoryChange.Survived:
|
||||||
@ -420,9 +438,7 @@ export class HistoryView extends View {
|
|||||||
console.log(`${key} ${this.sourceResolver.getPhaseNameById(key)}`);
|
console.log(`${key} ${this.sourceResolver.getPhaseNameById(key)}`);
|
||||||
const phaseHistory = this.phaseIdToHistory.get(key);
|
const phaseHistory = this.phaseIdToHistory.get(key);
|
||||||
for (const record of phaseHistory.nodeIdToRecord.values()) {
|
for (const record of phaseHistory.nodeIdToRecord.values()) {
|
||||||
const changes = Array.from(record.changes.values()).sort()
|
console.log(record.toString(), record.node);
|
||||||
.map(i => HistoryChange[i]).join(", ");
|
|
||||||
console.log(`[${changes}] `, record.node);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -444,6 +460,13 @@ export class PhaseHistory {
|
|||||||
}
|
}
|
||||||
this.nodeIdToRecord.get(key).addChange(change);
|
this.nodeIdToRecord.get(key).addChange(change);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public hasChanges(): boolean {
|
||||||
|
for (const record of this.nodeIdToRecord.values()) {
|
||||||
|
if (record.hasChanges()) return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class HistoryRecord {
|
export class HistoryRecord {
|
||||||
@ -458,10 +481,20 @@ export class HistoryRecord {
|
|||||||
public addChange(change: HistoryChange): void {
|
public addChange(change: HistoryChange): void {
|
||||||
this.changes.add(change);
|
this.changes.add(change);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public hasChanges(): boolean {
|
||||||
|
return this.changes.size > 1 ||
|
||||||
|
(this.changes.size == 1 && !this.changes.has(HistoryChange.Current));
|
||||||
|
}
|
||||||
|
|
||||||
|
public toString(): string {
|
||||||
|
return Array.from(this.changes.values()).sort().map(i => HistoryChange[i]).join(", ");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum HistoryChange {
|
export enum HistoryChange {
|
||||||
Current,
|
Current,
|
||||||
|
Emerged,
|
||||||
Lowered,
|
Lowered,
|
||||||
InplaceUpdated,
|
InplaceUpdated,
|
||||||
Removed,
|
Removed,
|
||||||
|
@ -16,12 +16,13 @@ import { TurboshaftGraph } from "../turboshaft-graph";
|
|||||||
import { Graph } from "../graph";
|
import { Graph } from "../graph";
|
||||||
import { TurboshaftGraphNode } from "../phases/turboshaft-graph-phase/turboshaft-graph-node";
|
import { TurboshaftGraphNode } from "../phases/turboshaft-graph-phase/turboshaft-graph-node";
|
||||||
import { GraphNode } from "../phases/graph-phase/graph-node";
|
import { GraphNode } from "../phases/graph-phase/graph-node";
|
||||||
|
import { SelectionStorage } from "../selection/selection-storage";
|
||||||
|
|
||||||
export abstract class MovableView<GraphType extends Graph | TurboshaftGraph> extends PhaseView {
|
export abstract class MovableView<GraphType extends Graph | TurboshaftGraph> extends PhaseView {
|
||||||
phaseName: string;
|
phaseName: string;
|
||||||
graph: GraphType;
|
graph: GraphType;
|
||||||
broker: SelectionBroker;
|
broker: SelectionBroker;
|
||||||
showPhaseByName: (name: string, selection: Set<any>) => void;
|
showPhaseByName: (name: string, selection: SelectionStorage) => void;
|
||||||
toolbox: HTMLElement;
|
toolbox: HTMLElement;
|
||||||
state: MovableViewState;
|
state: MovableViewState;
|
||||||
nodeSelectionHandler: NodeSelectionHandler & ClearableHandler;
|
nodeSelectionHandler: NodeSelectionHandler & ClearableHandler;
|
||||||
@ -35,7 +36,8 @@ export abstract class MovableView<GraphType extends Graph | TurboshaftGraph> ext
|
|||||||
public abstract svgKeyDown(): void;
|
public abstract svgKeyDown(): void;
|
||||||
|
|
||||||
constructor(idOrContainer: string | HTMLElement, broker: SelectionBroker,
|
constructor(idOrContainer: string | HTMLElement, broker: SelectionBroker,
|
||||||
showPhaseByName: (name: string) => void, toolbox: HTMLElement) {
|
showPhaseByName: (name: string, selection: SelectionStorage) => void,
|
||||||
|
toolbox: HTMLElement) {
|
||||||
super(idOrContainer);
|
super(idOrContainer);
|
||||||
this.broker = broker;
|
this.broker = broker;
|
||||||
this.showPhaseByName = showPhaseByName;
|
this.showPhaseByName = showPhaseByName;
|
||||||
|
@ -258,10 +258,10 @@ export abstract class TextView extends PhaseView {
|
|||||||
& ClearableHandler {
|
& ClearableHandler {
|
||||||
const view = this;
|
const view = this;
|
||||||
return {
|
return {
|
||||||
select: function (instructionIds: Array<string>, selected: boolean) {
|
select: function (instructionIds: Array<number>, selected: boolean) {
|
||||||
view.registerAllocationSelection.select(instructionIds, selected);
|
view.registerAllocationSelection.select(instructionIds, selected);
|
||||||
view.updateSelection();
|
view.updateSelection();
|
||||||
view.broker.broadcastInstructionSelect(null, [instructionIds], selected);
|
view.broker.broadcastInstructionSelect(null, instructionIds, selected);
|
||||||
},
|
},
|
||||||
clear: function () {
|
clear: function () {
|
||||||
view.registerAllocationSelection.clear();
|
view.registerAllocationSelection.clear();
|
||||||
|
@ -35,7 +35,8 @@ export class TurboshaftGraphView extends MovableView<TurboshaftGraph> {
|
|||||||
blockDrag: d3.DragBehavior<any, TurboshaftGraphBlock, TurboshaftGraphBlock>;
|
blockDrag: d3.DragBehavior<any, TurboshaftGraphBlock, TurboshaftGraphBlock>;
|
||||||
|
|
||||||
constructor(idOrContainer: string | HTMLElement, broker: SelectionBroker,
|
constructor(idOrContainer: string | HTMLElement, broker: SelectionBroker,
|
||||||
showPhaseByName: (name: string) => void, toolbox: HTMLElement) {
|
showPhaseByName: (name: string, selection: SelectionStorage) => void,
|
||||||
|
toolbox: HTMLElement) {
|
||||||
super(idOrContainer, broker, showPhaseByName, toolbox);
|
super(idOrContainer, broker, showPhaseByName, toolbox);
|
||||||
|
|
||||||
this.state.selection = new SelectionMap(node => node.identifier());
|
this.state.selection = new SelectionMap(node => node.identifier());
|
||||||
@ -155,7 +156,12 @@ export class TurboshaftGraphView extends MovableView<TurboshaftGraph> {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 85: // 'u'
|
case 85: // 'u'
|
||||||
this.collapseUnusedBlocks();
|
this.collapseUnusedBlocks(this.state.selection.selection.values());
|
||||||
|
break;
|
||||||
|
case 89: // 'y'
|
||||||
|
const node = this.graph.nodeMap[this.hoveredNodeIdentifier];
|
||||||
|
if (!node) return;
|
||||||
|
this.collapseUnusedBlocks([node]);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
eventHandled = false;
|
eventHandled = false;
|
||||||
@ -735,18 +741,22 @@ export class TurboshaftGraphView extends MovableView<TurboshaftGraph> {
|
|||||||
this.updateGraphVisibility();
|
this.updateGraphVisibility();
|
||||||
}
|
}
|
||||||
|
|
||||||
private collapseUnusedBlocks(): void {
|
private collapseUnusedBlocks(usedNodes: Iterable<TurboshaftGraphNode>): void {
|
||||||
const node = this.graph.nodeMap[this.hoveredNodeIdentifier];
|
const usedBlocks = new Set<TurboshaftGraphBlock>();
|
||||||
if (!node) return;
|
for (const node of usedNodes) {
|
||||||
|
usedBlocks.add(node.block);
|
||||||
|
|
||||||
const usedBlocks = new Set<TurboshaftGraphBlock>([node.block]);
|
for (const input of node.inputs) {
|
||||||
for (const input of node.inputs) {
|
usedBlocks.add(input.source.block);
|
||||||
usedBlocks.add(input.source.block);
|
}
|
||||||
}
|
|
||||||
for (const output of node.outputs) {
|
for (const output of node.outputs) {
|
||||||
usedBlocks.add(output.target.block);
|
usedBlocks.add(output.target.block);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (usedBlocks.size == 0) return;
|
||||||
|
|
||||||
for (const block of this.graph.blockMap) {
|
for (const block of this.graph.blockMap) {
|
||||||
block.collapsed = !usedBlocks.has(block);
|
block.collapsed = !usedBlocks.has(block);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user