diff --git a/python/dune/perftool/compile.py b/python/dune/perftool/compile.py index 9726396aa250cac3fe49ed3845bf833cf261877b..70d7a8a1ed580420f391143af592a351f59414b1 100644 --- a/python/dune/perftool/compile.py +++ b/python/dune/perftool/compile.py @@ -82,6 +82,9 @@ def compile_form(): kernels = generate_localoperator(form) # TODO insert sophisticated analysis/feedback loops here + if get_option("interactive"): + from dune.perftool.interactive import start_interactive_session + start_interactive_session(kernels) from dune.perftool.pdelab.localoperator import generate_localoperator_file generate_localoperator_file(kernels) diff --git a/python/dune/perftool/interactive.py b/python/dune/perftool/interactive.py new file mode 100644 index 0000000000000000000000000000000000000000..c58d69e82825fc4889554af28704d08ff4419e30 --- /dev/null +++ b/python/dune/perftool/interactive.py @@ -0,0 +1,122 @@ +from __future__ import print_function +from functools import partial + +# Use the builtin 'input' in python2 and 'raw_input' in python3 +try: + input = raw_input +except: + pass + + +def clear(): + import os + os.system('cls' if os.name == 'nt' else 'clear') + + +def kernel_name(v): + first = None + if v[1] == "residual": + first = "alpha" + if v[1] == "jacobian": + first = "jacobian" + assert first + + second = None + if v[0] == "cell": + second = "volume" + if v[0] == "exterior_facet": + second = "boundary" + if v[0] == "interior_facet": + second = "skeleton" + assert second + + return "{}_{}".format(first, second) + + +def show_kernel(kernel): + clear() + print(kernel) + print("Press any key to return") + input() + return kernel + + +def choose_transformation(which, kernel): + choice = None + while choice != "q": + clear() + keymap = {} + print("Choose one of the following transformations to apply to {}:\n".format(kernel_name(which))) + + print("Transformations:") + from dune.perftool.loopy.transformations import get_loopy_transformations + for i, v in enumerate(get_loopy_transformations().values()): + print(" {}) {}".format(chr(ord('a') + i), v.name)) + if v.description: + print(" {}".format(v.description)) + keymap[chr(ord('a') + i)] = v + + print("\n q) Return to kernel options") + print("\nYour choice:") + + choice = input().lower() + try: + kernel = keymap[choice](kernel) + except KeyError: + pass + + return kernel + + +def optimize_kernel(which, kernels): + kernel = kernels[which] + choice = None + + while choice != "q": + clear() + print("Optimizing kernel {}:\n".format(kernel_name(which))) + + print("Available options:") + print(" a) Show the loopy kernel") + print(" b) Apply loopy transformation") + + print("\n q) Return to the kernel overview") + print("\nYour choice:") + + choice = input().lower() + try: + kernel = {'a': show_kernel, + 'b': partial(choose_transformation, which) + }[choice](kernel) + except KeyError: + pass + + kernels[which] = kernel + + +def kernel_choice(kernels): + choice = None + while choice != "q": + clear() + print("The following kernels are in the input. Pick one to optimize:") + + keymap = {} + for i, k in enumerate(kernels.keys()): + print(" {}) {}".format(chr(ord('a') + i), kernel_name(k))) + keymap[chr(ord('a') + i)] = partial(optimize_kernel, k) + + print("\n q) End this interactive session and proceed to code generation") + + print("\nYour choice: ") + choice = input().lower() + try: + keymap[choice](kernels) + except KeyError: + pass + + +def start_interactive_session(kernels): + clear() + print("Welcome to the dune-perftool interactive mode!\n") + + kernel_choice(kernels) diff --git a/python/dune/perftool/loopy/transformations/__init__.py b/python/dune/perftool/loopy/transformations/__init__.py new file mode 100644 index 0000000000000000000000000000000000000000..db43b04544a9d0852d9181843e2a0fd554ac2bb6 --- /dev/null +++ b/python/dune/perftool/loopy/transformations/__init__.py @@ -0,0 +1,35 @@ +""" Infrastructure for loopy transformations. +These are registered to list them in interactive mode +""" + +_loopy_trafo_registry = {} + + +def get_loopy_transformations(): + return _loopy_trafo_registry + + +class LoopyTransformationWrapper(object): + def __init__(self, f, name=None, description=""): + self.func = f + self.name = name + self.description = description + + assert name + assert name not in _loopy_trafo_registry + + _loopy_trafo_registry[name] = self + + def __call__(self, kernel): + return self.func(kernel) + + +def loopy_transformation(_positional_arg=None, **kwargs): + assert not _positional_arg + return lambda f: LoopyTransformationWrapper(f, **kwargs) + + +# Just for debugging purposes we add an identity transformation here. +@loopy_transformation(name="identity", description='''Does not change the kernel. Proof of concept implementation''') +def _identity(kernel): + return kernel diff --git a/python/dune/perftool/options.py b/python/dune/perftool/options.py index db59bb9e927de59f0a7e98aa2768f210aab1aa64..4816ce8ecf5b2be5e4f6fd139b1fa79f4efa61d8 100644 --- a/python/dune/perftool/options.py +++ b/python/dune/perftool/options.py @@ -12,6 +12,7 @@ def get_form_compiler_arguments(): parser.add_argument("--operator-file", type=str, help="The filename for the generated local operator header") parser.add_argument('--version', action='version', version='%(prog)s 0.1') parser.add_argument("--numerical-jacobian", action="store_true", help="use numerical jacobians (only makes sense, if uflpdelab for some reason fails to generate analytic jacobians)") + parser.add_argument("--interactive", action="store_true", help="whether the optimization process should be guided interactively (also useful for debugging)") # These are the options that I deemed necessary in uflpdelab # parser.add_argument("--param-class-file", type=str, help="The filename for the generated parameter class header") diff --git a/python/dune/perftool/ufl/transformations/__init__.py b/python/dune/perftool/ufl/transformations/__init__.py index d67f112bc8a1b8a34d7a6887d76cd548e4801449..006e44cd0de8c31ef9d7fbd99fec8a14f8e62007 100644 --- a/python/dune/perftool/ufl/transformations/__init__.py +++ b/python/dune/perftool/ufl/transformations/__init__.py @@ -1,6 +1,6 @@ """ Define the general infrastructure for debuggable UFL transformations""" -class TransformationWrapper(object): +class UFLTransformationWrapper(object): def __init__(self, func, **kwargs): # Store the decorated function self.func = func @@ -63,7 +63,7 @@ def ufl_transformation(_positional_arg=None, **kwargs): """ A decorator for ufl transformations. It allows us to output the result if needed. """ assert not _positional_arg - return lambda f: TransformationWrapper(f, **kwargs) + return lambda f: UFLTransformationWrapper(f, **kwargs) @ufl_transformation(name="print", printBefore=False) def print_expression(e): @@ -72,13 +72,13 @@ def print_expression(e): def transform_integral(integral, trafo): from ufl import Integral assert isinstance(integral, Integral) - assert isinstance(trafo, TransformationWrapper) + assert isinstance(trafo, UFLTransformationWrapper) return integral.reconstruct(integrand=trafo(integral.integrand())) def transform_form(form, trafo): from ufl import Form assert isinstance(form, Form) - assert isinstance(trafo, TransformationWrapper) + assert isinstance(trafo, UFLTransformationWrapper) return Form([transform_integral(i, trafo) for i in form.integrals()])