diff --git a/dace/sdfg/work_depth_analysis/op_in_helpers.py b/dace/sdfg/work_depth_analysis/op_in_helpers.py deleted file mode 100644 index c7c17741de..0000000000 --- a/dace/sdfg/work_depth_analysis/op_in_helpers.py +++ /dev/null @@ -1,78 +0,0 @@ -# Copyright 2019-2023 ETH Zurich and the DaCe authors. All rights reserved. -""" Contains class CacheLineTracker which keeps track of all arrays of an SDFG and their cache line position. -Further, contains class AccessStack which which corresponds to the stack used to compute the stack distance. """ - -from dace.data import Array - -class CacheLineTracker: - - def __init__(self, L) -> None: - self.array_info = {} - self.start_lines = {} - self.next_free_line = 0 - self.L = L - - def add_array(self, name: str, a: Array): - if name not in self.start_lines: - # new array encountered - self.array_info[name] = a - self.start_lines[name] = self.next_free_line - # increase next_free_line - self.next_free_line += (a.total_size * a.dtype.bytes + self.L - 1) // self.L # ceil division - - def cache_line_id(self, name: str, access: [int]): - arr = self.array_info[name] - one_d_index = 0 - for dim in range(len(access)): - i = access[dim] - one_d_index += (i + arr.offset[dim]) * arr.strides[dim] - - # divide by L to get the cache line id - return self.start_lines[name] + (one_d_index * arr.dtype.bytes) // self.L - - -class Node: - - def __init__(self, val: int, n=None) -> None: - self.v = val - self.next = n - - -class AccessStack: - """ A stack of cache line ids. For each memory access, we search the corresponding cache line id - in the stack, report its distance and move it to the top of the stack. If the id was not found, - we report a distance of -1. """ - - def __init__(self) -> None: - self.top = None - - def touch(self, id): - - curr = self.top - prev = None - found = False - distance = 0 - while curr is not None: - # check if we found id - if curr.v == id: - # take curr node out - if prev is not None: - prev.next = curr.next - curr.next = self.top - self.top = curr - - found = True - break - - # iterate further - prev = curr - curr = curr.next - distance += 1 - - if not found: - # we accessed this cache line for the first time ever - self.top = Node(id, self.top) - distance = -1 - - return distance - diff --git a/dace/sdfg/work_depth_analysis/operational_intensity.py b/dace/sdfg/work_depth_analysis/operational_intensity.py deleted file mode 100644 index af94c7f924..0000000000 --- a/dace/sdfg/work_depth_analysis/operational_intensity.py +++ /dev/null @@ -1,1004 +0,0 @@ -# Copyright 2019-2023 ETH Zurich and the DaCe authors. All rights reserved. -""" Analyses the operational intensity of an input SDFG. Can be used as a Python script -or from the VS Code extension. """ - -""" -Plan: -- For each memory access, we need to figure out its cache line and then we compute its stack distance. -- For that we model the actual stack, where we push all the memory acesses (What do we push exactly? -Cache line ids?? check typescript implementation for that information.) -- How do we know which array maps to which cache line? - Idea: for each new array encountered, just assume that it is cache line aligned and starts - at the next free cache line. TODO: check if this is how it usually behaves. Or are arrays - aligned further, like base address % x == 0 for some x bigger than cache line size? -- It is also important that we take data types into account for each array. -- For each mem access we increase the miss counter if stack distance > C(apacity) or it it is a -compulsory miss. Then, in the end we know how many bytes are transferred to cache. It is: - num_misses * L(ine size in bytes) - -- Parameters to our analysis are - - input SDFG - - C(ache capacity) - - L(ine size) -""" - - - - - - - - - - -import argparse -from collections import deque -from dace.sdfg import nodes as nd, propagation, InterstateEdge -from dace import SDFG, SDFGState, dtypes, int64 -from dace.subsets import Range -from typing import Tuple, Dict -import os -import sympy as sp -from copy import deepcopy -from dace.libraries.blas import MatMul -from dace.libraries.standard import Reduce, Transpose -from dace.symbolic import pystr_to_symbolic -import ast -import astunparse -import warnings - -from dace.sdfg.work_depth_analysis.helpers import get_uuid, find_loop_guards_tails_exits -from dace.sdfg.work_depth_analysis.assumptions import parse_assumptions -from dace.transformation.passes.symbol_ssa import StrictSymbolSSA -from dace.transformation.pass_pipeline import FixedPointPipeline - -from dace.data import Array -from dace.sdfg.work_depth_analysis.op_in_helpers import CacheLineTracker, AccessStack - -def get_array_size_symbols(sdfg): - """ - Returns all symbols that appear isolated in shapes of the SDFG's arrays. - These symbols can then be assumed to be positive. - - :note: This only works if a symbol appears in isolation, i.e. array A[N]. - If we have A[N+1], we cannot assume N to be positive. - :param sdfg: The SDFG in which it searches for symbols. - :return: A set containing symbols which we can assume to be positive. - """ - symbols = set() - for _, _, arr in sdfg.arrays_recursive(): - for s in arr.shape: - if isinstance(s, sp.Symbol): - symbols.add(s) - return symbols - - -def symeval(val, symbols): - """ - Takes a sympy expression and substitutes its symbols according to a dict { old_symbol: new_symbol}. - - :param val: The expression we are updating. - :param symbols: Dictionary of key value pairs { old_symbol: new_symbol}. - """ - first_replacement = {pystr_to_symbolic(k): pystr_to_symbolic('__REPLSYM_' + k) for k in symbols.keys()} - second_replacement = {pystr_to_symbolic('__REPLSYM_' + k): v for k, v in symbols.items()} - return sp.simplify(val.subs(first_replacement).subs(second_replacement)) - - -def evaluate_symbols(base, new): - result = {} - for k, v in new.items(): - result[k] = symeval(v, base) - return result - - -def count_work_matmul(node, symbols, state): - A_memlet = next(e for e in state.in_edges(node) if e.dst_conn == '_a') - B_memlet = next(e for e in state.in_edges(node) if e.dst_conn == '_b') - C_memlet = next(e for e in state.out_edges(node) if e.src_conn == '_c') - result = 2 # Multiply, add - # Batch - if len(C_memlet.data.subset) == 3: - result *= symeval(C_memlet.data.subset.size()[0], symbols) - # M*N - result *= symeval(C_memlet.data.subset.size()[-2], symbols) - result *= symeval(C_memlet.data.subset.size()[-1], symbols) - # K - result *= symeval(A_memlet.data.subset.size()[-1], symbols) - return sp.sympify(result) - - -def count_depth_matmul(node, symbols, state): - # optimal depth of a matrix multiplication is O(log(size of shared dimension)): - A_memlet = next(e for e in state.in_edges(node) if e.dst_conn == '_a') - size_shared_dimension = symeval(A_memlet.data.subset.size()[-1], symbols) - return bigo(sp.log(size_shared_dimension)) - - -def count_work_reduce(node, symbols, state): - result = 0 - if node.wcr is not None: - result += count_arithmetic_ops_code(node.wcr) - in_memlet = None - in_edges = state.in_edges(node) - if in_edges is not None and len(in_edges) == 1: - in_memlet = in_edges[0] - if in_memlet is not None and in_memlet.data.volume is not None: - result *= in_memlet.data.volume - else: - result = 0 - return sp.sympify(result) - - -def count_depth_reduce(node, symbols, state): - # optimal depth of reduction is log of the work - return bigo(sp.log(count_work_reduce(node, symbols, state))) - - -LIBNODES_TO_WORK = { - MatMul: count_work_matmul, - Transpose: lambda *args: 0, - Reduce: count_work_reduce, -} - -LIBNODES_TO_DEPTH = { - MatMul: count_depth_matmul, - Transpose: lambda *args: 0, - Reduce: count_depth_reduce, -} - -bigo = sp.Function('bigo') -PYFUNC_TO_ARITHMETICS = { - 'float': 0, - 'dace.float64': 0, - 'dace.int64': 0, - 'math.exp': 1, - 'exp': 1, - 'math.tanh': 1, - 'sin': 1, - 'cos': 1, - 'tanh': 1, - 'math.sqrt': 1, - 'sqrt': 1, - 'atan2:': 1, - 'min': 0, - 'max': 0, - 'ceiling': 0, - 'floor': 0, - 'abs': 0 -} - - -class ArithmeticCounter(ast.NodeVisitor): - - def __init__(self): - self.count = 0 - - def visit_BinOp(self, node): - if isinstance(node.op, ast.MatMult): - raise NotImplementedError('MatMult op count requires shape ' - 'inference') - self.count += 1 - return self.generic_visit(node) - - def visit_UnaryOp(self, node): - self.count += 1 - return self.generic_visit(node) - - def visit_Call(self, node): - fname = astunparse.unparse(node.func)[:-1] - if fname not in PYFUNC_TO_ARITHMETICS: - print( - 'WARNING: Unrecognized python function "%s". If this is a type conversion, like "dace.float64", then this is fine.' - % fname) - return self.generic_visit(node) - self.count += PYFUNC_TO_ARITHMETICS[fname] - return self.generic_visit(node) - - def visit_AugAssign(self, node): - return self.visit_BinOp(node) - - def visit_For(self, node): - raise NotImplementedError - - def visit_While(self, node): - raise NotImplementedError - - -def count_arithmetic_ops_code(code): - ctr = ArithmeticCounter() - if isinstance(code, (tuple, list)): - for stmt in code: - ctr.visit(stmt) - elif isinstance(code, str): - ctr.visit(ast.parse(code)) - else: - ctr.visit(code) - return ctr.count - - -class DepthCounter(ast.NodeVisitor): - # so far this is identical to the ArithmeticCounter above. - def __init__(self): - self.count = 0 - - def visit_BinOp(self, node): - if isinstance(node.op, ast.MatMult): - raise NotImplementedError('MatMult op count requires shape ' - 'inference') - self.count += 1 - return self.generic_visit(node) - - def visit_UnaryOp(self, node): - self.count += 1 - return self.generic_visit(node) - - def visit_Call(self, node): - fname = astunparse.unparse(node.func)[:-1] - if fname not in PYFUNC_TO_ARITHMETICS: - print( - 'WARNING: Unrecognized python function "%s". If this is a type conversion, like "dace.float64", then this is fine.' - % fname) - return self.generic_visit(node) - self.count += PYFUNC_TO_ARITHMETICS[fname] - return self.generic_visit(node) - - def visit_AugAssign(self, node): - return self.visit_BinOp(node) - - def visit_For(self, node): - raise NotImplementedError - - def visit_While(self, node): - raise NotImplementedError - - -def count_depth_code(code): - # so far this is the same as the work counter, since work = depth for each tasklet, as we can't assume any parallelism - ctr = ArithmeticCounter() - if isinstance(code, (tuple, list)): - for stmt in code: - ctr.visit(stmt) - elif isinstance(code, str): - ctr.visit(ast.parse(code)) - else: - ctr.visit(code) - return ctr.count - - -def tasklet_work(tasklet_node, state): - if tasklet_node.code.language == dtypes.Language.CPP: - # simplified work analysis for CPP tasklets. - for oedge in state.out_edges(tasklet_node): - return oedge.data.num_accesses - elif tasklet_node.code.language == dtypes.Language.Python: - return count_arithmetic_ops_code(tasklet_node.code.code) - else: - # other languages not implemented, count whole tasklet as work of 1 - warnings.warn('Work of tasklets only properly analyzed for Python or CPP. For all other ' - 'languages work = 1 will be counted for each tasklet.') - return 1 - - -def tasklet_depth(tasklet_node, state): - if tasklet_node.code.language == dtypes.Language.CPP: - # For now we simply take depth == work for CPP tasklets. - for oedge in state.out_edges(tasklet_node): - return oedge.data.num_accesses - if tasklet_node.code.language == dtypes.Language.Python: - return count_depth_code(tasklet_node.code.code) - else: - # other languages not implemented, count whole tasklet as work of 1 - warnings.warn('Depth of tasklets only properly analyzed for Python code. For all other ' - 'languages depth = 1 will be counted for each tasklet.') - return 1 - - -def get_tasklet_work(node, state): - return sp.sympify(tasklet_work(node, state)), sp.sympify(-1) - - -def get_tasklet_work_depth(node, state): - return sp.sympify(tasklet_work(node, state)), sp.sympify(tasklet_depth(node, state)) - - -def get_tasklet_avg_par(node, state): - return sp.sympify(tasklet_work(node, state)), sp.sympify(tasklet_depth(node, state)) - - -def update_value_map(old, new): - # add new assignments to old - old.update({k: v for k, v in new.items() if k not in old}) - # check for conflicts: - for k, v in new.items(): - if k in old and old[k] != v: - # conflict detected --> forget this mapping completely - old.pop(k) - - -def do_initial_subs(w, d, eq, subs1): - """ - Calls subs three times for the give (w)ork and (d)epth values. - """ - return sp.simplify(w.subs(eq[0]).subs(eq[1]).subs(subs1)), sp.simplify(d.subs(eq[0]).subs(eq[1]).subs(subs1)) - - -def sdfg_op_in_OLD(sdfg: SDFG, op_in_map: Dict[str, Tuple[sp.Expr, sp.Expr]], analyze_tasklet, symbols: Dict[str, str], - detailed_analysis: bool, equality_subs: Tuple[Dict[str, sp.Symbol], Dict[str, sp.Expr]], - subs1: Dict[str, sp.Expr]) -> Tuple[sp.Expr, sp.Expr]: - """ - Analyze the work and depth of a given SDFG. - First we determine the work and depth of each state. Then we break loops in the state machine, such that we get a DAG. - Lastly, we compute the path with most work and the path with the most depth in order to get the total work depth. - - :param sdfg: The SDFG to analyze. - :param op_in_map: Dictionary which will save the result. - :param analyze_tasklet: Function used to analyze tasklet nodes. - :param symbols: A dictionary mapping local nested SDFG symbols to global symbols. - :param detailed_analysis: If True, detailed analysis gets used. For each branch, we keep track of its condition - and work depth values for both branches. If False, the worst-case branch is taken. Discouraged to use on bigger SDFGs, - as computation time sky-rockets, since expression can became HUGE (depending on number of branches etc.). - :param equality_subs: Substitution dict taking care of the equality assumptions. - :param subs1: First substitution dict for greater/lesser assumptions. - :return: A tuple containing the work and depth of the SDFG. - """ - - # First determine the work and depth of each state individually. - # Keep track of the work and depth for each state in a dictionary, where work and depth are multiplied by the number - # of times the state will be executed. - state_depths: Dict[SDFGState, sp.Expr] = {} - state_works: Dict[SDFGState, sp.Expr] = {} - for state in sdfg.nodes(): - state_work, state_depth = state_op_in_OLD(state, op_in_map, analyze_tasklet, symbols, detailed_analysis, - equality_subs, subs1) - - # Substitutions for state_work and state_depth already performed, but state.executions needs to be subs'd now. - state_work = sp.simplify(state_work * - state.executions.subs(equality_subs[0]).subs(equality_subs[1]).subs(subs1)) - state_depth = sp.simplify(state_depth * - state.executions.subs(equality_subs[0]).subs(equality_subs[1]).subs(subs1)) - - state_works[state], state_depths[state] = state_work, state_depth - op_in_map[get_uuid(state)] = (state_works[state], state_depths[state]) - - # Prepare the SDFG for a depth analysis by breaking loops. This removes the edge between the last loop state and - # the guard, and instead places an edge between the last loop state and the exit state. - # This transforms the state machine into a DAG. Hence, we can find the "heaviest" and "deepest" paths in linear time. - # Additionally, construct a dummy exit state and connect every state that has no outgoing edges to it. - - # identify all loops in the SDFG - nodes_oNodes_exits = find_loop_guards_tails_exits(sdfg._nx) - - # Now we need to go over each triple (node, oNode, exits). For each triple, we - # - remove edge (oNode, node), i.e. the backward edge - # - for all exits e, add edge (oNode, e). This edge may already exist - # - remove edge from node to exit (if present, i.e. while-do loop) - # - This ensures that every node with > 1 outgoing edge is a branch guard - # - useful for detailed anaylsis. - for node, oNode, exits in nodes_oNodes_exits: - sdfg.remove_edge(sdfg.edges_between(oNode, node)[0]) - for e in exits: - if len(sdfg.edges_between(oNode, e)) == 0: - # no edge there yet - sdfg.add_edge(oNode, e, InterstateEdge()) - if len(sdfg.edges_between(node, e)) > 0: - # edge present --> remove it - sdfg.remove_edge(sdfg.edges_between(node, e)[0]) - - # add a dummy exit to the SDFG, such that each path ends there. - dummy_exit = sdfg.add_state('dummy_exit') - for state in sdfg.nodes(): - if len(sdfg.out_edges(state)) == 0 and state != dummy_exit: - sdfg.add_edge(state, dummy_exit, InterstateEdge()) - - # These two dicts save the current length of the "heaviest", resp. "deepest", paths at each state. - work_map: Dict[SDFGState, sp.Expr] = {} - depth_map: Dict[SDFGState, sp.Expr] = {} - # Keeps track of assignments done on InterstateEdges. - state_value_map: Dict[SDFGState, Dict[sp.Symbol, sp.Symbol]] = {} - # The dummy state has 0 work and depth. - state_depths[dummy_exit] = sp.sympify(0) - state_works[dummy_exit] = sp.sympify(0) - - # Perform a BFS traversal of the state machine and calculate the maximum work / depth at each state. Only advance to - # the next state in the BFS if all incoming edges have been visited, to ensure the maximum work / depth expressions - # have been calculated. - traversal_q = deque() - traversal_q.append((sdfg.start_state, sp.sympify(0), sp.sympify(0), None, [], [], {})) - visited = set() - - while traversal_q: - state, depth, work, ie, condition_stack, common_subexpr_stack, value_map = traversal_q.popleft() - - if ie is not None: - visited.add(ie) - - if state in state_value_map: - # update value map: - update_value_map(state_value_map[state], value_map) - else: - state_value_map[state] = value_map - - # ignore assignments such as tmp=x[0], as those do not give much information. - value_map = {k: v for k, v in state_value_map[state].items() if '[' not in k and '[' not in v} - n_depth = sp.simplify((depth + state_depths[state]).subs(value_map)) - n_work = sp.simplify((work + state_works[state]).subs(value_map)) - - # If we are analysing average parallelism, we don't search "heaviest" and "deepest" paths separately, but we want one - # single path with the least average parallelsim (of all paths with more than 0 work). - if analyze_tasklet == get_tasklet_avg_par: - if state in depth_map: # this means we have already visited this state before - cse = common_subexpr_stack.pop() - # if current path has 0 depth (--> 0 work as well), we don't do anything. - if n_depth != 0: - # check if we need to update the work and depth of the current state - # we update if avg parallelism of new incoming path is less than current avg parallelism - if depth_map[state] == 0: - # old value was divided by zero --> we take new value anyway - depth_map[state] = cse[1] + n_depth - work_map[state] = cse[0] + n_work - else: - old_avg_par = (cse[0] + work_map[state]) / (cse[1] + depth_map[state]) - new_avg_par = (cse[0] + n_work) / (cse[1] + n_depth) - # we take either old work/depth or new work/depth (or both if we cannot determine which one is greater) - depth_map[state] = cse[1] + sp.Piecewise((n_depth, sp.simplify(new_avg_par < old_avg_par)), - (depth_map[state], True)) - work_map[state] = cse[0] + sp.Piecewise((n_work, sp.simplify(new_avg_par < old_avg_par)), - (work_map[state], True)) - else: - depth_map[state] = n_depth - work_map[state] = n_work - else: - # search heaviest and deepest path separately - if state in depth_map: # and consequently also in work_map - # This cse value would appear in both arguments of the Max. Hence, for performance reasons, - # we pull it out of the Max expression. - # Example: We do cse + Max(a, b) instead of Max(cse + a, cse + b). - # This increases performance drastically, expecially since we avoid nesting Max expressions - # for cases where cse itself contains Max operators. - cse = common_subexpr_stack.pop() - if detailed_analysis: - # This MAX should be covered in the more detailed analysis - cond = condition_stack.pop() - work_map[state] = cse[0] + sp.Piecewise((work_map[state], sp.Not(cond)), (n_work, cond)) - depth_map[state] = cse[1] + sp.Piecewise((depth_map[state], sp.Not(cond)), (n_depth, cond)) - else: - work_map[state] = cse[0] + sp.Max(work_map[state], n_work) - depth_map[state] = cse[1] + sp.Max(depth_map[state], n_depth) - else: - depth_map[state] = n_depth - work_map[state] = n_work - - out_edges = sdfg.out_edges(state) - # only advance after all incoming edges were visited (meaning that current work depth values of state are final). - if any(iedge not in visited for iedge in sdfg.in_edges(state)): - pass - else: - for oedge in out_edges: - if len(out_edges) > 1: - # It is important to copy these stacks. Else both branches operate on the same stack. - # state is a branch guard --> save condition on stack - new_cond_stack = list(condition_stack) - new_cond_stack.append(oedge.data.condition_sympy()) - # same for common_subexr_stack - new_cse_stack = list(common_subexpr_stack) - new_cse_stack.append((work_map[state], depth_map[state])) - # same for value_map - new_value_map = dict(state_value_map[state]) - new_value_map.update({sp.Symbol(k): sp.Symbol(v) for k, v in oedge.data.assignments.items()}) - traversal_q.append((oedge.dst, 0, 0, oedge, new_cond_stack, new_cse_stack, new_value_map)) - else: - value_map.update(oedge.data.assignments) - traversal_q.append((oedge.dst, depth_map[state], work_map[state], oedge, condition_stack, - common_subexpr_stack, value_map)) - - try: - max_depth = depth_map[dummy_exit] - max_work = work_map[dummy_exit] - except KeyError: - # If we get a KeyError above, this means that the traversal never reached the dummy_exit state. - # This happens if the loops were not properly detected and broken. - raise Exception( - 'Analysis failed, since not all loops got detected. It may help to use more structured loop constructs.') - - sdfg_result = (max_work, max_depth) - op_in_map[get_uuid(sdfg)] = sdfg_result - return sdfg_result - - -def scope_op_in_OLD(state: SDFGState, - op_in_map: Dict[str, sp.Expr], - analyze_tasklet, - symbols: Dict[str, str], - detailed_analysis: bool, - equality_subs: Tuple[Dict[str, sp.Symbol], Dict[str, sp.Expr]], - subs1: Dict[str, sp.Expr], - entry: nd.EntryNode = None) -> Tuple[sp.Expr, sp.Expr]: - """ - Analyze the work and depth of a scope. - This works by traversing through the scope analyzing the work and depth of each encountered node. - Depending on what kind of node we encounter, we do the following: - - EntryNode: Recursively analyze work depth of scope. - - Tasklet: use analyze_tasklet to get work depth of tasklet node. - - NestedSDFG: After translating its local symbols to global symbols, we analyze the nested SDFG recursively. - - LibraryNode: Library nodes are analyzed with special functions depending on their type. - Work inside a state can simply be summed up, but for the depth we need to find the longest path. Since dataflow is a DAG, - this can be done in linear time by traversing the graph in topological order. - - :param state: The state in which the scope to analyze is contained. - :param op_in_map: Dictionary saving the final result for each SDFG element. - :param analyze_tasklet: Function used to analyze tasklets. Either analyzes just work, work and depth or average parallelism. - :param symbols: A dictionary mapping local nested SDFG symbols to global symbols. - :param detailed_analysis: If True, detailed analysis gets used. For each branch, we keep track of its condition - and work depth values for both branches. If False, the worst-case branch is taken. Discouraged to use on bigger SDFGs, - as computation time sky-rockets, since expression can became HUGE (depending on number of branches etc.). - :param equality_subs: Substitution dict taking care of the equality assumptions. - :param subs1: First substitution dict for greater/lesser assumptions. - :param entry: The entry node of the scope to analyze. If None, the entire state is analyzed. - :return: A tuple containing the work and depth of the scope. - """ - - # find the work and depth of each node - # for maps and nested SDFG, we do it recursively - work = sp.sympify(0) - max_depth = sp.sympify(0) - scope_nodes = state.scope_children()[entry] - scope_exit = None if entry is None else state.exit_node(entry) - for node in scope_nodes: - # add node to map - op_in_map[get_uuid(node, state)] = (sp.sympify(0), sp.sympify(0)) - if isinstance(node, nd.EntryNode): - # If the scope contains an entry node, we need to recursively analyze the sub-scope of the entry node first. - # The resulting work/depth are summarized into the entry node - s_work, s_depth = scope_op_in_OLD(state, op_in_map, analyze_tasklet, symbols, detailed_analysis, - equality_subs, subs1, node) - s_work, s_depth = do_initial_subs(s_work, s_depth, equality_subs, subs1) - # add up work for whole state, but also save work for this sub-scope scope in op_in_map - work += s_work - op_in_map[get_uuid(node, state)] = (s_work, s_depth) - elif node == scope_exit: - # don't do anything for exit nodes, everthing handled already in the corresponding entry node. - pass - elif isinstance(node, nd.Tasklet): - # add up work for whole state, but also save work for this node in op_in_map - # t_work, t_depth = analyze_tasklet(node, state) - - # analyze the memory accesses of this tasklet and whether they hit in cache or not - print('tasklet') - t_work, t_depth = sp.sympify(100), sp.sympify(100) - - - # check if tasklet has any outgoing wcr edges - for e in state.out_edges(node): - if e.data.wcr is not None: - t_work += count_arithmetic_ops_code(e.data.wcr) - t_work, t_depth = do_initial_subs(t_work, t_depth, equality_subs, subs1) - work += t_work - op_in_map[get_uuid(node, state)] = (t_work, t_depth) - elif isinstance(node, nd.NestedSDFG): - # keep track of nested symbols: "symbols" maps local nested SDFG symbols to global symbols. - # We only want global symbols in our final work depth expressions. - nested_syms = {} - nested_syms.update(symbols) - nested_syms.update(evaluate_symbols(symbols, node.symbol_mapping)) - # Nested SDFGs are recursively analyzed first. - nsdfg_work, nsdfg_depth = sdfg_op_in_OLD(node.sdfg, op_in_map, analyze_tasklet, nested_syms, - detailed_analysis, equality_subs, subs1) - - nsdfg_work, nsdfg_depth = do_initial_subs(nsdfg_work, nsdfg_depth, equality_subs, subs1) - # add up work for whole state, but also save work for this nested SDFG in op_in_map - work += nsdfg_work - op_in_map[get_uuid(node, state)] = (nsdfg_work, nsdfg_depth) - elif isinstance(node, nd.LibraryNode): - try: - lib_node_work = LIBNODES_TO_WORK[type(node)](node, symbols, state) - except KeyError: - # add a symbol to the top level sdfg, such that the user can define it in the extension - top_level_sdfg = state.parent - # TODO: This symbol should now appear in the VS code extension in the SDFG analysis tab, - # such that the user can define its value. But it doesn't... - # How to achieve this? - top_level_sdfg.add_symbol(f'{node.name}_work', int64) - lib_node_work = sp.Symbol(f'{node.name}_work', positive=True) - lib_node_depth = sp.sympify(-1) # not analyzed - if analyze_tasklet != get_tasklet_work: - # we are analyzing depth - try: - lib_node_depth = LIBNODES_TO_DEPTH[type(node)](node, symbols, state) - except KeyError: - top_level_sdfg = state.parent - top_level_sdfg.add_symbol(f'{node.name}_depth', int64) - lib_node_depth = sp.Symbol(f'{node.name}_depth', positive=True) - lib_node_work, lib_node_depth = do_initial_subs(lib_node_work, lib_node_depth, equality_subs, subs1) - work += lib_node_work - op_in_map[get_uuid(node, state)] = (lib_node_work, lib_node_depth) - - if entry is not None: - # If the scope being analyzed is a map, multiply the work by the number of iterations of the map. - if isinstance(entry, nd.MapEntry): - nmap: nd.Map = entry.map - range: Range = nmap.range - n_exec = range.num_elements_exact() - work = sp.simplify(work * n_exec.subs(equality_subs[0]).subs(equality_subs[1]).subs(subs1)) - else: - print('WARNING: Only Map scopes are supported in work analysis for now. Assuming 1 iteration.') - - # Work inside a state can simply be summed up. But now we need to find the depth of a state (i.e. longest path). - # Since dataflow graph is a DAG, this can be done in linear time. - max_depth = sp.sympify(0) - # only do this if we are analyzing depth - if analyze_tasklet == get_tasklet_work_depth or analyze_tasklet == get_tasklet_avg_par: - # Calculate the maximum depth of the scope by finding the 'deepest' path from the source to the sink. This is done by - # a traversal in topological order, where each node propagates its current max depth for all incoming paths. - traversal_q = deque() - visited = set() - # find all starting nodes - if entry: - # the entry is the starting node - traversal_q.append((entry, sp.sympify(0), None)) - else: - for node in scope_nodes: - if len(state.in_edges(node)) == 0: - # This node is a start node of the traversal - traversal_q.append((node, sp.sympify(0), None)) - # this map keeps track of the length of the longest path ending at each state so far seen. - depth_map = {} - wcr_depth_map = {} - while traversal_q: - node, in_depth, in_edge = traversal_q.popleft() - - if in_edge is not None: - visited.add(in_edge) - - n_depth = sp.simplify(in_depth + op_in_map[get_uuid(node, state)][1]) - - if node in depth_map: - depth_map[node] = sp.Max(depth_map[node], n_depth) - else: - depth_map[node] = n_depth - - out_edges = state.out_edges(node) - # Only advance to next node, if all incoming edges have been visited or the current node is the entry (aka starting node). - # If the current node is the exit of the scope, we stop, such that we don't leave the scope. - if (all(iedge in visited for iedge in state.in_edges(node)) or node == entry) and node != scope_exit: - # If we encounter a nested map, we must not analyze its contents (as they have already been recursively analyzed). - # Hence, we continue from the outgoing edges of the corresponding exit. - if isinstance(node, nd.EntryNode) and node != entry: - exit_node = state.exit_node(node) - # replace out_edges with the out_edges of the scope exit node - out_edges = state.out_edges(exit_node) - for oedge in out_edges: - # check for wcr - wcr_depth = sp.sympify(0) - if oedge.data.wcr is not None: - # This division gives us the number of writes to each single memory location, which is the depth - # as these need to be sequential (without assumptions on HW etc). - wcr_depth = oedge.data.volume / oedge.data.subset.num_elements() - if get_uuid(node, state) in wcr_depth_map: - # max - wcr_depth_map[get_uuid(node, state)] = sp.Max(wcr_depth_map[get_uuid(node, state)], - wcr_depth) - else: - wcr_depth_map[get_uuid(node, state)] = wcr_depth - # We do not need to propagate the wcr_depth to MapExits, since else this will result in depth N + 1 for Maps of range N. - wcr_depth = wcr_depth if not isinstance(oedge.dst, nd.MapExit) else sp.sympify(0) - - # only append if it's actually new information - # this e.g. helps for huge nested SDFGs with lots of inputs/outputs inside a map scope - append = True - for n, d, _ in traversal_q: - if oedge.dst == n and depth_map[node] + wcr_depth == d: - append = False - break - if append: - traversal_q.append((oedge.dst, depth_map[node] + wcr_depth, oedge)) - else: - visited.add(oedge) - if len(out_edges) == 0 or node == scope_exit: - # We have reached an end node --> update max_depth - max_depth = sp.Max(max_depth, depth_map[node]) - - for uuid in wcr_depth_map: - op_in_map[uuid] = (op_in_map[uuid][0], op_in_map[uuid][1] + wcr_depth_map[uuid]) - # summarise work / depth of the whole scope in the dictionary - scope_result = (work, max_depth) - op_in_map[get_uuid(state)] = scope_result - return scope_result - - -def state_op_in_OLD(state: SDFGState, op_in_map: Dict[str, sp.Expr], analyze_tasklet, symbols, detailed_analysis, - equality_subs, subs1) -> Tuple[sp.Expr, sp.Expr]: - """ - Analyze the work and depth of a state. - - :param state: The state to analyze. - :param op_in_map: The result will be saved to this map. - :param analyze_tasklet: Function used to analyze tasklet nodes. - :param symbols: A dictionary mapping local nested SDFG symbols to global symbols. - :param detailed_analysis: If True, detailed analysis gets used. For each branch, we keep track of its condition - and work depth values for both branches. If False, the worst-case branch is taken. Discouraged to use on bigger SDFGs, - as computation time sky-rockets, since expression can became HUGE (depending on number of branches etc.). - :param equality_subs: Substitution dict taking care of the equality assumptions. - :param subs1: First substitution dict for greater/lesser assumptions. - :return: A tuple containing the work and depth of the state. - """ - work, depth = scope_op_in_OLD(state, op_in_map, analyze_tasklet, symbols, detailed_analysis, equality_subs, subs1, - None) - return work, depth - - -def analyze_sdfg(sdfg: SDFG, op_in_map: Dict[str, sp.Expr], analyze_tasklet, assumptions: [str], - detailed_analysis: bool) -> None: - """ - Analyze a given SDFG. We can either analyze work, work and depth or average parallelism. - - :note: SDFGs should have split interstate edges. This means there should be no interstate edges containing both a - condition and an assignment. - :param sdfg: The SDFG to analyze. - :param op_in_map: Dictionary of SDFG elements to (work, depth) tuples. Result will be saved in here. - :param analyze_tasklet: Function used to analyze tasklet nodes. Analyzes either just work, work and depth or average parallelism. - :param assumptions: List of strings. Each string corresponds to one assumption for some symbol, e.g. 'N>5'. - :param detailed_analysis: If True, detailed analysis gets used. For each branch, we keep track of its condition - and work depth values for both branches. If False, the worst-case branch is taken. Discouraged to use on bigger SDFGs, - as computation time sky-rockets, since expression can became HUGE (depending on number of branches etc.). - """ - - # deepcopy such that original sdfg not changed - sdfg = deepcopy(sdfg) - - # apply SSA pass - pipeline = FixedPointPipeline([StrictSymbolSSA()]) - pipeline.apply_pass(sdfg, {}) - - array_symbols = get_array_size_symbols(sdfg) - # parse assumptions - equality_subs, all_subs = parse_assumptions(assumptions if assumptions is not None else [], array_symbols) - - # Run state propagation for all SDFGs recursively. This is necessary to determine the number of times each state - # will be executed, or to determine upper bounds for that number (such as in the case of branching) - for sd in sdfg.all_sdfgs_recursive(): - propagation.propagate_states(sd, concretize_dynamic_unbounded=True) - - # Analyze the work and depth of the SDFG. - symbols = {} - sdfg_op_in_OLD(sdfg, op_in_map, analyze_tasklet, symbols, detailed_analysis, equality_subs, - all_subs[0][0] if len(all_subs) > 0 else {}) - - for k, (v_w, v_d) in op_in_map.items(): - # The symeval replaces nested SDFG symbols with their global counterparts. - v_w, v_d = do_subs(v_w, v_d, all_subs) - v_w = symeval(v_w, symbols) - v_d = symeval(v_d, symbols) - op_in_map[k] = (v_w, v_d) - - -def do_subs(work, depth, all_subs): - """ - Handles all substitutions beyond the equality substitutions and the first substitution. - :param work: Some work expression. - :param depth: Some depth expression. - :param all_subs: List of substitution pairs to perform. - :return: Work depth expressions after doing all substitutions. - """ - # first do subs2 of first sub - # then do all the remaining subs - subs2 = all_subs[0][1] if len(all_subs) > 0 else {} - work, depth = sp.simplify(sp.sympify(work).subs(subs2)), sp.simplify(sp.sympify(depth).subs(subs2)) - for i in range(1, len(all_subs)): - subs1, subs2 = all_subs[i] - work, depth = sp.simplify(work.subs(subs1)), sp.simplify(depth.subs(subs1)) - work, depth = sp.simplify(work.subs(subs2)), sp.simplify(depth.subs(subs2)) - return work, depth - - - - - - - - - - - - -def update_mapping(map, mapping): - # update the map params and return False - # if all iterations exhausted, return True - # always increase the last one, if it is exhausted, increase the next one and so forth - map_exhausted = True - for p, range in zip(map.params[::-1], map.range[::-1]): - curr_value = mapping[p] - if curr_value < range[1]: - # update this value and we done - mapping[p] = curr_value + range[2] - map_exhausted = False - break - else: - # set current param to start again and continue - mapping[p] = range[0] - return map_exhausted - - - -def map_op_in(state: SDFGState, op_in_map: Dict[str, sp.Expr], entry, mapping, stack, clt, C): - # we are inside a map --> we need to iterate over the map range and check each memory access. - for p, range in zip(entry.map.params, entry.map.range): - # map each map iteration variable to its start - mapping[p] = range[0] - map_misses = 0 - while True: - # do analysis of map contents - map_misses += scope_op_in(state, op_in_map, mapping, stack, clt, C, entry) - - if update_mapping(entry.map, mapping): - break - return map_misses - - -def scope_op_in(state: SDFGState, op_in_map: Dict[str, sp.Expr], mapping, stack: AccessStack, clt: CacheLineTracker, C, entry=None): - # find the work and depth of each node - # for maps and nested SDFG, we do it recursively - scope_misses = 0 - scope_nodes = state.scope_children()[entry] - for node in scope_nodes: - # add node to map - op_in_map[get_uuid(node, state)] = 0 - if isinstance(node, nd.EntryNode): - # If the scope contains an entry node, we need to recursively analyze the sub-scope of the entry node first. - # The resulting work/depth are summarized into the entry node - map_misses = map_op_in(state, op_in_map, node, mapping, stack, clt, C) - # add up work for whole state, but also save work for this sub-scope scope in op_in_map - op_in_map[get_uuid(node, state)] = map_misses - scope_misses += map_misses - elif isinstance(node, nd.Tasklet): - # add up work for whole state, but also save work for this node in op_in_map - tasklet_misses = 0 - # analyze the memory accesses of this tasklet and whether they hit in cache or not - for e in state.in_edges(node): - if e.data.data in clt.array_info: - line_id = clt.cache_line_id(e.data.data, [x[0].subs(mapping) for x in e.data.subset.ranges]) - dist = stack.touch(line_id) - tasklet_misses += 1 if dist > C or dist == -1 else 0 - for e in state.out_edges(node): - if e.data.data in clt.array_info: - line_id = clt.cache_line_id(e.data.data, [x[0].subs(mapping) for x in e.data.subset.ranges]) - dist = stack.touch(line_id) - tasklet_misses += 1 if dist > C or dist == -1 else 0 - - # TODO: wcr edges. - scope_misses += tasklet_misses - op_in_map[get_uuid(node, state)] = tasklet_misses - elif isinstance(node, nd.NestedSDFG): - pass - # keep track of nested symbols: "symbols" maps local nested SDFG symbols to global symbols. - # We only want global symbols in our final work depth expressions. - # nested_syms = {} - # nested_syms.update(symbols) - # nested_syms.update(evaluate_symbols(symbols, node.symbol_mapping)) - # Nested SDFGs are recursively analyzed first. - nsdfg_misses = sdfg_op_in(node.sdfg, op_in_map, mapping, stack, clt, C) - - # nsdfg_work, nsdfg_depth = do_initial_subs(nsdfg_work, nsdfg_depth, equality_subs, subs1) - # add up work for whole state, but also save work for this nested SDFG in op_in_map - scope_misses += nsdfg_misses - op_in_map[get_uuid(node, state)] = nsdfg_misses - elif isinstance(node, nd.LibraryNode): - pass - # try: - # lib_node_work = LIBNODES_TO_WORK[type(node)](node, symbols, state) - # except KeyError: - # # add a symbol to the top level sdfg, such that the user can define it in the extension - # top_level_sdfg = state.parent - # # TODO: This symbol should now appear in the VS code extension in the SDFG analysis tab, - # # such that the user can define its value. But it doesn't... - # # How to achieve this? - # top_level_sdfg.add_symbol(f'{node.name}_work', int64) - # lib_node_work = sp.Symbol(f'{node.name}_work', positive=True) - # lib_node_depth = sp.sympify(-1) # not analyzed - # if analyze_tasklet != get_tasklet_work: - # # we are analyzing depth - # try: - # lib_node_depth = LIBNODES_TO_DEPTH[type(node)](node, symbols, state) - # except KeyError: - # top_level_sdfg = state.parent - # top_level_sdfg.add_symbol(f'{node.name}_depth', int64) - # lib_node_depth = sp.Symbol(f'{node.name}_depth', positive=True) - # lib_node_work, lib_node_depth = do_initial_subs(lib_node_work, lib_node_depth, equality_subs, subs1) - # work += lib_node_work - # op_in_map[get_uuid(node, state)] = (lib_node_work, lib_node_depth) - op_in_map[get_uuid(state)] = scope_misses - return scope_misses - -def sdfg_op_in(sdfg: SDFG, op_in_map: Dict[str, Tuple[sp.Expr, sp.Expr]], mapping, stack, clt, C): - # traverse this SDFG's states - curr_state = sdfg.start_state - total_misses = 0 - while True: - total_misses += scope_op_in(curr_state, op_in_map, mapping, stack, clt, C) - - if len(sdfg.out_edges(curr_state)) == 0: - # we reached the end state --> stop - break - else: - # take first edge with True condition - found = False - for e in sdfg.out_edges(curr_state): - if e.data.is_unconditional() or e.data.condition_sympy().subs(mapping) == True: - # save e's assignments in mapping and update curr_state - # replace values first with mapping, then update mapping - mapping.update({k: sp.sympify(v).subs(mapping) for k, v in e.data.assignments.items()}) - curr_state = e.dst - found = True - break - if not found: - print('WARNING: state has outgoing edges, but no condition of them can be' - 'evaluated as True and hence the analysis ends. ') - break - # traverse further - op_in_map[get_uuid(sdfg)] = total_misses - -def analyze_sdfg_op_in(sdfg: SDFG, op_in_map: Dict[str, sp.Expr], C, L): - # do some top level stuff - stack = AccessStack() - clt = CacheLineTracker(L) - for _, name, arr in sdfg.arrays_recursive(): - if isinstance(arr, Array): - if name in clt.array_info: - print('WARNING: This array name was already seen!!! Two arrays with the same name in the SDFG.') - clt.add_array(name, arr) - mapping = {} - - sdfg_op_in(sdfg, op_in_map, mapping, stack, clt, C) - - # now we have number of misses --> multiply each by L - for k, v in op_in_map.items(): - op_in_map[k] = v * L - - - - - - -################################################################################ -# Utility functions for running the analysis from the command line ############# -################################################################################ - - -def main() -> None: - - parser = argparse.ArgumentParser('operational_intensity', - usage='python operational_intensity.py [-h] filename', - description='Analyze the operational_intensity of an SDFG.') - - parser.add_argument('filename', type=str, help='The SDFG file to analyze.') - parser.add_argument('C', type=str, help='Cache size in bytes') - parser.add_argument('L', type=str, help='Cache line size in bytes') - - args = parser.parse_args() - - if not os.path.exists(args.filename): - print(args.filename, 'does not exist.') - exit() - - sdfg = SDFG.from_file(args.filename) - op_in_map = {} - analyze_sdfg_op_in(sdfg, op_in_map, int(args.C), int(args.L)) - - for k, v, in op_in_map.items(): - op_in_map[k] = str(sp.simplify(v)) - - result_whole_sdfg = op_in_map[get_uuid(sdfg)] - - print(80 * '-') - print("Bytes transferred:\t", result_whole_sdfg) - print(80 * '-') - - -if __name__ == '__main__': - main() - - - - - -