From c0b79d47273adbbf5b63645184ba9a13dfb618f3 Mon Sep 17 00:00:00 2001 From: Philipp Schaad Date: Fri, 19 Apr 2024 16:38:09 +0200 Subject: [PATCH] Fix some of the overlays for hierarchical CFGs --- src/overlays/memory_location_overlay.ts | 54 +++--- src/overlays/memory_volume_overlay.ts | 229 +++++++++++++----------- 2 files changed, 154 insertions(+), 129 deletions(-) diff --git a/src/overlays/memory_location_overlay.ts b/src/overlays/memory_location_overlay.ts index 2a6b3a7d..afea9f1f 100644 --- a/src/overlays/memory_location_overlay.ts +++ b/src/overlays/memory_location_overlay.ts @@ -4,9 +4,12 @@ import { DagreGraph, Point2D, SimpleRect } from '../index'; import { SDFGRenderer } from '../renderer/renderer'; import { AccessNode, + ControlFlowBlock, + ControlFlowRegion, NestedSDFG, SDFGElement, - SDFGNode + SDFGNode, + State } from '../renderer/renderer_elements'; import { SDFV } from '../sdfv'; import { KELLY_COLORS } from '../utils/utils'; @@ -180,15 +183,17 @@ export class MemoryLocationOverlay extends GenericSdfgOverlay { let storageType = sdfgArray?.attributes?.storage; let originalType: StorageType | null = null; - if (storageType) { - if (storageType === StorageType.Default) { - const schedule = - MemoryLocationOverlay.recursiveFindScopeSchedule(node); - const derivedStorageType = SCOPEDEFAULT_STORAGE.get(schedule); - if (derivedStorageType) { - originalType = storageType; - storageType = derivedStorageType; - } + + if (!storageType) + storageType = StorageType.Default; + + if (storageType === StorageType.Default) { + const schedule = + MemoryLocationOverlay.recursiveFindScopeSchedule(node); + const derivedStorageType = SCOPEDEFAULT_STORAGE.get(schedule); + if (derivedStorageType) { + originalType = storageType; + storageType = derivedStorageType; } } return { @@ -219,32 +224,33 @@ export class MemoryLocationOverlay extends GenericSdfgOverlay { node.shade(this.renderer, ctx, '#' + color); } - public recursivelyShadeSdfg( + public recursivelyShadeCFG( graph: DagreGraph, ctx: CanvasRenderingContext2D, ppp: number, visibleRect: SimpleRect ): void { - // First go over visible states, skipping invisible ones. We traverse - // inside to shade memory nodes wherever applicable. + // First go over visible control flow blocks, skipping invisible ones. + // We traverse inside to shade memory nodes wherever applicable. graph.nodes().forEach(v => { - const state = graph.node(v); + const block: ControlFlowBlock = graph.node(v); // If the node's invisible, we skip it. - if ((ctx as any).lod && !state.intersect( + if ((ctx as any).lod && !block.intersect( visibleRect.x, visibleRect.y, visibleRect.w, visibleRect.h )) return; if (((ctx as any).lod && (ppp >= SDFV.STATE_LOD || - state.width / ppp <= SDFV.STATE_LOD)) || - state.data.state.attributes.is_collapsed) { - // The state is collapsed or invisible, so we don't need to + block.width / ppp <= SDFV.STATE_LOD)) || + block.data.state?.attributes.is_collapsed || + block.data.block?.attributes.is_collapsed) { + // The block is collapsed or invisible, so we don't need to // traverse its insides. return; - } else { - const stateGraph = state.data.graph; + } else if (block instanceof State) { + const stateGraph = block.data.graph; if (stateGraph) { stateGraph.nodes().forEach((v: any) => { const node = stateGraph.node(v); @@ -257,7 +263,7 @@ export class MemoryLocationOverlay extends GenericSdfgOverlay { if (node instanceof NestedSDFG && node.attributes().sdfg && node.attributes().sdfg.type !== 'SDFGShell') { - this.recursivelyShadeSdfg( + this.recursivelyShadeCFG( node.data.graph, ctx, ppp, visibleRect ); } else if (node instanceof AccessNode) { @@ -265,6 +271,10 @@ export class MemoryLocationOverlay extends GenericSdfgOverlay { } }); } + } else if (block instanceof ControlFlowRegion) { + this.recursivelyShadeCFG( + block.data.graph, ctx, ppp, visibleRect + ); } }); } @@ -275,7 +285,7 @@ export class MemoryLocationOverlay extends GenericSdfgOverlay { const context = this.renderer.get_context(); const visibleRect = this.renderer.get_visible_rect(); if (graph && ppp !== undefined && context && visibleRect) - this.recursivelyShadeSdfg(graph, context, ppp, visibleRect); + this.recursivelyShadeCFG(graph, context, ppp, visibleRect); } public on_mouse_event( diff --git a/src/overlays/memory_volume_overlay.ts b/src/overlays/memory_volume_overlay.ts index d51c49c3..d105ae5b 100644 --- a/src/overlays/memory_volume_overlay.ts +++ b/src/overlays/memory_volume_overlay.ts @@ -3,6 +3,8 @@ import { DagreGraph, Point2D, SimpleRect, SymbolMap } from '../index'; import { SDFGRenderer } from '../renderer/renderer'; import { + ControlFlowBlock, + ControlFlowRegion, Edge, NestedSDFG, SDFGElement, @@ -24,7 +26,7 @@ export class MemoryVolumeOverlay extends GenericSdfgOverlay { this.refresh(); } - public clear_cached_volume_values(): void { + public clearCachedVolumeValues(): void { this.renderer.doForAllGraphElements((_group, info, obj: any) => { if (obj.data) { if (obj.data.volume !== undefined) @@ -33,90 +35,96 @@ export class MemoryVolumeOverlay extends GenericSdfgOverlay { }); } - public calculate_volume_edge( + public calculateVolumeEdge( edge: Edge, - symbol_map: SymbolMap, - volume_values: number[] + symbolMap: SymbolMap, + volumes: number[] ): number | undefined { - let volume_string = undefined; + let volumeString = undefined; if (edge.data && edge.data.attributes) { - volume_string = edge.data.attributes.volume; - if (volume_string !== undefined) { - volume_string = volume_string.replace(/\*\*/g, '^'); - volume_string = volume_string.replace(/ceiling/g, 'ceil'); + volumeString = edge.data.attributes.volume; + if (volumeString !== undefined) { + volumeString = volumeString.replace(/\*\*/g, '^'); + volumeString = volumeString.replace(/ceiling/g, 'ceil'); } } let volume = undefined; - if (volume_string !== undefined) + if (volumeString !== undefined) volume = this.symbol_resolver.parse_symbol_expression( - volume_string, - symbol_map + volumeString, + symbolMap ); edge.data.volume = volume; if (volume !== undefined && volume > 0) - volume_values.push(volume); + volumes.push(volume); return volume; } - public calculate_volume_graph( + public calculateVolumeGraph( g: DagreGraph, - symbol_map: SymbolMap, - volume_values: number[] + symbolMaps: SymbolMap, + volumes: number[] ): void { g.nodes().forEach((v: string) => { - const state = g.node(v); - const state_graph = state.data.graph; - if (state_graph) { - state_graph.edges().forEach((e: number) => { - const edge = state_graph.edge(e); - if (edge instanceof Edge) - this.calculate_volume_edge( - edge, - symbol_map, - volume_values - ); - }); - - state_graph.nodes().forEach((v: number) => { - const node = state_graph.node(v); - if (node instanceof NestedSDFG) { - const nested_symbols_map: SymbolMap = {}; - const mapping = node.data.node.attributes.symbol_mapping ?? {}; - // Translate the symbol mappings for the nested SDFG - // based on the mapping described on the node. - Object.keys(mapping).forEach((symbol) => { - nested_symbols_map[symbol] = - this.symbol_resolver.parse_symbol_expression( - mapping[symbol], - symbol_map - ); - }); - // Merge in the parent mappings. - Object.keys(symbol_map).forEach((symbol) => { - if (!(symbol in nested_symbols_map)) - nested_symbols_map[symbol] = symbol_map[symbol]; - }); - - this.calculate_volume_graph( - node.data.graph, - nested_symbols_map, - volume_values - ); - } - }); + const block: ControlFlowBlock = g.node(v); + if (block instanceof State) { + const stateGraph = block.data.graph; + if (stateGraph) { + stateGraph.edges().forEach((e: number) => { + const edge = stateGraph.edge(e); + if (edge instanceof Edge) + this.calculateVolumeEdge( + edge, + symbolMaps, + volumes + ); + }); + + stateGraph.nodes().forEach((v: number) => { + const node = stateGraph.node(v); + if (node instanceof NestedSDFG) { + const nested_symbols_map: SymbolMap = {}; + const mapping = node.data.node.attributes.symbol_mapping ?? {}; + // Translate the symbol mappings for the nested SDFG + // based on the mapping described on the node. + Object.keys(mapping).forEach((symbol) => { + nested_symbols_map[symbol] = + this.symbol_resolver.parse_symbol_expression( + mapping[symbol], + symbolMaps + ); + }); + // Merge in the parent mappings. + Object.keys(symbolMaps).forEach((symbol) => { + if (!(symbol in nested_symbols_map)) + nested_symbols_map[symbol] = symbolMaps[symbol]; + }); + + this.calculateVolumeGraph( + node.data.graph, + nested_symbols_map, + volumes + ); + } + }); + } + } else if (block instanceof ControlFlowRegion) { + this.calculateVolumeGraph( + block.data.graph, symbolMaps, volumes + ); } }); } - public recalculate_volume_values(graph: DagreGraph): void { + public recalculateVolumeValues(graph: DagreGraph): void { this.heatmap_scale_center = 5; this.heatmap_hist_buckets = []; const volume_values: number[] = []; - this.calculate_volume_graph( + this.calculateVolumeGraph( graph, this.symbol_resolver.get_symbol_value_map(), volume_values @@ -129,15 +137,15 @@ export class MemoryVolumeOverlay extends GenericSdfgOverlay { } public refresh(): void { - this.clear_cached_volume_values(); + this.clearCachedVolumeValues(); const graph = this.renderer.get_graph(); if (graph) - this.recalculate_volume_values(graph); + this.recalculateVolumeValues(graph); this.renderer.draw_async(); } - public shade_edge(edge: Edge, ctx: CanvasRenderingContext2D): void { + public shadeEdge(edge: Edge, ctx: CanvasRenderingContext2D): void { const volume = edge.data.volume; if (volume !== undefined) { // Only draw positive volumes. @@ -153,62 +161,69 @@ export class MemoryVolumeOverlay extends GenericSdfgOverlay { } } - public recursively_shade_sdfg( + public recursivelyShadeCFG( graph: DagreGraph, ctx: CanvasRenderingContext2D, ppp: number, - visible_rect: SimpleRect + visibleRect: SimpleRect ): void { graph.nodes().forEach(v => { - const state: State = graph.node(v); + const block: ControlFlowBlock = graph.node(v); // If we're zoomed out enough that the contents aren't visible, we // skip the state. if ((ctx as any).lod && ( - ppp >= SDFV.STATE_LOD || state.width / ppp < SDFV.STATE_LOD + ppp >= SDFV.STATE_LOD || block.width / ppp < SDFV.STATE_LOD )) return; // If the node's invisible, we skip it. - if ((ctx as any).lod && !state.intersect( - visible_rect.x, visible_rect.y, - visible_rect.w, visible_rect.h - )) + if ((ctx as any).lod && !block.intersect( + visibleRect.x, visibleRect.y, + visibleRect.w, visibleRect.h + ) || block.data.state?.attributes.is_collapsed || + block.data.block?.attributes.is_collapsed) return; - const state_graph = state.data.graph; - if (state_graph && !state.data.state.attributes.is_collapsed) { - state_graph.nodes().forEach((v: string) => { - const node: SDFGNode = state_graph.node(v); - - // Skip the node if it's not visible. - if ((ctx as any).lod && !node.intersect( - visible_rect.x, visible_rect.y, - visible_rect.w, visible_rect.h - )) - return; - - // If we're zoomed out enough that the node's contents - // aren't visible or the node is collapsed, we skip it. - if (node.data.node.attributes.is_collapsed || - ((ctx as any).lod && ppp >= SDFV.NODE_LOD)) - return; - - if (node instanceof NestedSDFG && - node.attributes().sdfg && - node.attributes().sdfg.type !== 'SDFGShell') - this.recursively_shade_sdfg( - node.data.graph, ctx, ppp, visible_rect - ); - }); - - state_graph.edges().forEach((e: any) => { - const edge: Edge = state_graph.edge(e); - - if ((ctx as any).lod && !edge.intersect(visible_rect.x, - visible_rect.y, visible_rect.w, visible_rect.h)) - return; - - this.shade_edge(edge, ctx); - }); + if (block instanceof State) { + const state_graph = block.data.graph; + if (state_graph) { + state_graph.nodes().forEach((v: string) => { + const node: SDFGNode = state_graph.node(v); + + // Skip the node if it's not visible. + if ((ctx as any).lod && !node.intersect( + visibleRect.x, visibleRect.y, + visibleRect.w, visibleRect.h + )) + return; + + // If we're zoomed out enough that the node's contents + // aren't visible or the node is collapsed, we skip it. + if (node.data.node.attributes.is_collapsed || + ((ctx as any).lod && ppp >= SDFV.NODE_LOD)) + return; + + if (node instanceof NestedSDFG && + node.attributes().sdfg && + node.attributes().sdfg.type !== 'SDFGShell') + this.recursivelyShadeCFG( + node.data.graph, ctx, ppp, visibleRect + ); + }); + + state_graph.edges().forEach((e: any) => { + const edge: Edge = state_graph.edge(e); + + if ((ctx as any).lod && !edge.intersect(visibleRect.x, + visibleRect.y, visibleRect.w, visibleRect.h)) + return; + + this.shadeEdge(edge, ctx); + }); + } + } else if (block instanceof ControlFlowRegion) { + this.recursivelyShadeCFG( + block.data.graph, ctx, ppp, visibleRect + ); } }); } @@ -219,7 +234,7 @@ export class MemoryVolumeOverlay extends GenericSdfgOverlay { const context = this.renderer.get_context(); const visible_rect = this.renderer.get_visible_rect(); if (graph && ppp !== undefined && context && visible_rect) - this.recursively_shade_sdfg(graph, context, ppp, visible_rect); + this.recursivelyShadeCFG(graph, context, ppp, visible_rect); } public on_mouse_event( @@ -242,8 +257,8 @@ export class MemoryVolumeOverlay extends GenericSdfgOverlay { () => { const graph = this.renderer.get_graph(); if (graph) { - this.clear_cached_volume_values(); - this.recalculate_volume_values(graph); + this.clearCachedVolumeValues(); + this.recalculateVolumeValues(graph); } } );