From 3928f8023a1a287e57a9d16184306915a038d1a6 Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Mon, 18 Sep 2017 15:06:17 +0200 Subject: [PATCH 01/11] yask: Fix Expression instantiation --- devito/yask/utils.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/devito/yask/utils.py b/devito/yask/utils.py index 3e3f515482..80dd016ac7 100644 --- a/devito/yask/utils.py +++ b/devito/yask/utils.py @@ -45,7 +45,7 @@ def make_grid_gets(expr): processed = Element(c.Statement(ccode(handle))) else: # Writing to a scalar temporary - processed = Expression(e.expr.func(lhs, rhs)) + processed = Expression(e.expr.func(lhs, rhs), dtype=e.dtype) mapper.update({e: processed}) From 2472e1d643879696b45a257f5a68e38bb224c2cd Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Mon, 18 Sep 2017 15:05:43 +0200 Subject: [PATCH 02/11] yask: Add test_increasing_multi_steps --- tests/test_yask.py | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/tests/test_yask.py b/tests/test_yask.py index 523624ed5d..df51ef523e 100644 --- a/tests/test_yask.py +++ b/tests/test_yask.py @@ -162,6 +162,20 @@ def test_increasing_halo_wo_ofs(self, space_order): assert all(np.all(u.data.with_halo[1, :, i, :] == 0) for i in range(space_order)) assert all(np.all(u.data.with_halo[1, :, :, i] == 0) for i in range(space_order)) + def test_increasing_multi_steps(self): + """ + Apply the trivial equation ``u[t+1,x,y,z] = u[t,x,y,z] + 1`` for 11 + timesteps and check that all grid domain values are equal to 11 within + ``u[1]`` and equal to 10 within ``u[0]``. + """ + u = TimeData(name='yu4D', shape=(8, 8, 8), dimensions=(x, y, z), space_order=0) + u.data.with_halo[:] = 0. + op = Operator(Eq(u.forward, u + 1.), subs={t.spacing: 1}) + op(yu4D=u, t=12) + assert 'run_solution' in str(op) + assert np.all(u.data[0] == 10.) + assert np.all(u.data[1] == 11.) + @pytest.mark.parametrize("space_order", [2]) def test_fixed_halo_w_ofs(self, space_order): """ From eff329f444f7f43f3f9dc0d4d217af6eeb222521 Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Tue, 19 Sep 2017 16:44:22 +0200 Subject: [PATCH 03/11] Avoid caching hits with FunctionFromPointer due to the way sympy.Symbols are cached, just based on the name. Now I'm also passing the pointer name --- devito/cgen_utils.py | 2 +- devito/dse/extended_sympy.py | 7 ++++--- devito/yask/utils.py | 6 +++--- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/devito/cgen_utils.py b/devito/cgen_utils.py index fe02d44309..01420d455b 100644 --- a/devito/cgen_utils.py +++ b/devito/cgen_utils.py @@ -145,7 +145,7 @@ def _print_FrozenExpr(self, expr): def _print_FunctionFromPointer(self, expr): indices = [self._print(i) for i in expr.params] - return "%s->%s(%s)" % (expr.pointer, expr.name, ', '.join(indices)) + return "%s->%s(%s)" % (expr.pointer, expr.function, ', '.join(indices)) def ccode(expr, **settings): diff --git a/devito/dse/extended_sympy.py b/devito/dse/extended_sympy.py index 036003d698..ac3a354703 100644 --- a/devito/dse/extended_sympy.py +++ b/devito/dse/extended_sympy.py @@ -54,14 +54,15 @@ class Add(sympy.Add, FrozenExpr): class FunctionFromPointer(sympy.Symbol): - def __new__(cls, name, pointer, params=None): - obj = sympy.Symbol.__new__(cls, name) + def __new__(cls, function, pointer, params=None): + obj = sympy.Symbol.__new__(cls, '%s->%s' % (pointer, function)) + obj.function = function obj.pointer = pointer obj.params = params or () return obj def __str__(self): - return "%s->%s(%s)" % (self.pointer, self.name, + return "%s->%s(%s)" % (self.pointer, self.function, ', '.join([str(i) for i in self.params])) __repr__ = __str__ diff --git a/devito/yask/utils.py b/devito/yask/utils.py index 80dd016ac7..bace9f95dc 100644 --- a/devito/yask/utils.py +++ b/devito/yask/utils.py @@ -21,14 +21,14 @@ def make_grid_accesses(node): """ def make_grid_gets(expr): + mapper = {} indexeds = retrieve_indexed(expr) data_carriers = [i for i in indexeds if i.base.function.from_YASK] for i in data_carriers: name = namespace['code-grid-name'](i.base.function.name) args = [INT(make_grid_gets(j)) for j in i.indices] - handle = make_sharedptr_funcall(namespace['code-grid-get'], args, name) - expr = expr.xreplace({i: handle}) - return expr + mapper[i] = make_sharedptr_funcall(namespace['code-grid-get'], args, name) + return expr.xreplace(mapper) mapper = {} for i, e in enumerate(FindNodes(Expression).visit(node)): From 3e36c631af665d4f9fc01f206263332d3ed40bd1 Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Tue, 19 Sep 2017 16:44:51 +0200 Subject: [PATCH 04/11] yask: Switch off temporarily EXTRA_MACROS=TRACE --- devito/yask/wrappers.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/devito/yask/wrappers.py b/devito/yask/wrappers.py index 4fbb44c86f..cee6400fdf 100644 --- a/devito/yask/wrappers.py +++ b/devito/yask/wrappers.py @@ -230,7 +230,7 @@ def __init__(self, name, yc_soln, domain): # JIT-compile it try: make(os.environ['YASK_HOME'], ['-j', 'YK_CXXOPT=-O0', - "EXTRA_MACROS=TRACE", + # "EXTRA_MACROS=TRACE", 'YK_BASE=%s' % str(name), 'stencil=%s' % yc_soln.get_name(), '-C', namespace['kernel-path'], 'api']) From 76aa636eecefc9de2709d2031c4dada8a52470f5 Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Mon, 18 Sep 2017 17:30:49 +0200 Subject: [PATCH 05/11] yask: Restructure wave acoustic test Seems to work -- at least, u.data is sane --- tests/test_yask.py | 135 +++++++++++++++++++++++++++++++-------------- 1 file changed, 94 insertions(+), 41 deletions(-) diff --git a/tests/test_yask.py b/tests/test_yask.py index df51ef523e..398b2c190c 100644 --- a/tests/test_yask.py +++ b/tests/test_yask.py @@ -6,13 +6,13 @@ pexpect = pytest.importorskip('yask_compiler') # Run only if YASK is available from devito import (Operator, DenseData, TimeData, PointData, - t, x, y, z, configuration, clear_cache, info) # noqa + time, t, x, y, z, configuration, clear_cache, info) # noqa from devito.dle import retrieve_iteration_tree # noqa from devito.yask.wrappers import YaskGrid, contexts # noqa # For the acoustic wave test from examples.seismic.acoustic import AcousticWaveSolver # noqa -from examples.seismic import demo_model, RickerSource, Receiver # noqa +from examples.seismic import demo_model, PointSource, RickerSource, Receiver # noqa pytestmark = pytest.mark.skipif(configuration['backend'] != 'yask', reason="'yask' wasn't selected as backend on startup") @@ -113,7 +113,7 @@ def test_data_arithmetic_nD(self, u): assert np.all(arr - u.data == -1.) -class TestOperatorExecution(object): +class TestOperatorSimple(object): """ Test execution of "toy" Operators through YASK. @@ -294,7 +294,7 @@ def test_irregular_write(self): assert all(np.all(u.data[1, :, :, i] == 3 - i) for i in range(4)) -class TestOperatorRealAcoustic(object): +class TestOperatorAcoustic(object): """ Test the acoustic wave model through YASK. @@ -307,51 +307,104 @@ class TestOperatorRealAcoustic(object): 'layers': {'preset': 'layers', 'ratio': 3}, } - @pytest.mark.parametrize('mkey, dimensions, time_order, space_order, nbpml', [ - # 3D tests with varying space orders - pytest.mark.xfail(('layers', (60, 70, 80), 2, 4, 10)), - pytest.mark.xfail(('layers', (60, 70, 80), 2, 8, 10)), - ]) - def test_acoustic(self, mkey, dimensions, time_order, space_order, nbpml): - t0 = 0.0 # Start time - tn = 500. # Final time - nrec = 130 # Number of receivers - - # Create model from preset - model = demo_model(spacing=[15. for _ in dimensions], - shape=dimensions, nbpml=nbpml, **(self.presets[mkey])) + @pytest.fixture + def model(self): + shape = (60, 70, 80) + nbpml = 10 + mkey = 'layers' + return demo_model(spacing=[15, 15, 15], shape=shape, nbpml=nbpml, + preset='layers', ratio=3) + @pytest.fixture + def time_params(self, model): # Derive timestepping from model spacing - dt = model.critical_dt * (1.73 if time_order == 4 else 1.0) + t0 = 0.0 # Start time + tn = 500. # Final time + dt = model.critical_dt nt = int(1 + (tn-t0) / dt) # Number of timesteps - time_values = np.linspace(t0, tn, nt) # Discretized time axis - + return t0, tn, nt + + @pytest.fixture + def m(self, model): + return model.m + + @pytest.fixture + def damp(self, model): + return model.damp + + @pytest.fixture + def u(self, model): + space_order = 4 + time_order = 2 + return TimeData(name='u', shape=model.shape_domain, dimensions=(x, y, z), + space_order=space_order, time_order=time_order) + + @pytest.fixture + def stencil(self, m, damp, u): + s = t.spacing + stencil = 1.0 / (2.0 * m + s * damp) * ( + 4.0 * m * u + (s * damp - 2.0 * m) * u.backward + + 2.0 * s**2 * u.laplace) + return stencil + + @pytest.fixture + def src(self, model, time_params): + dt = model.critical_dt + time_values = np.linspace(*time_params) # Discretized time axis # Define source geometry (center of domain, just below surface) src = RickerSource(name='src', ndim=model.dim, f0=0.01, time=time_values) src.coordinates.data[0, :] = np.array(model.domain_size) * .5 src.coordinates.data[0, -1] = 30. + return src - # Define receiver geometry (same as source, but spread across x) - rec = Receiver(name='nrec', ntime=nt, npoint=nrec, ndim=model.dim) + @pytest.fixture + def rec(self, model, time_params, src): + nrec = 130 # Number of receivers + t0, tn, nt = time_params + rec = Receiver(name='rec', ntime=nt, npoint=nrec, ndim=model.dim) rec.coordinates.data[:, 0] = np.linspace(0., model.domain_size[0], num=nrec) rec.coordinates.data[:, 1:] = src.coordinates.data[0, 1:] + return rec - # Create solver object to provide relevant operators - solver = AcousticWaveSolver(model, source=src, receiver=rec, - time_order=time_order, - space_order=space_order) - - # Create adjoint receiver symbol - srca = Receiver(name='srca', ntime=solver.source.nt, - coordinates=solver.source.coordinates.data) - - # Run forward and adjoint operators - rec, _, _ = solver.forward(save=False) - solver.adjoint(rec=rec, srca=srca) - - # Adjoint test: Verify matches closely - term1 = np.dot(srca.data.reshape(-1), solver.source.data) - term2 = np.linalg.norm(rec.data) ** 2 - info(': %f, : %f, difference: %12.12f, ratio: %f' - % (term1, term2, term1 - term2, term1 / term2)) - assert np.isclose(term1, term2, rtol=1.e-5) + @pytest.fixture + def subs(self, model, u): + dt = model.critical_dt + return dict([(t.spacing, dt)] + [(time.spacing, dt)] + + [(i.spacing, model.get_spacing()[j]) for i, j + in zip(u.indices[1:], range(len(model.shape)))]) + + def test_acoustic_wo_src_wo_rec(self, model, stencil, subs, m, damp, u): + """ + Test that the acoustic wave equation runs without crashing in absence + of sources and receivers. + """ + u.data[:] = 0.0 + eqn = [Eq(u.forward, stencil)] + op = Operator(eqn, subs=subs) + op.apply(u=u, m=m, damp=damp, t=10) + + def test_acoustic_w_src_wo_rec(self, model, stencil, subs, m, damp, u, src): + """ + Test that the acoustic wave equation runs without crashing in absence + of receivers. + """ + dt = model.critical_dt + u.data[:] = 0.0 + eqns = [Eq(u.forward, stencil)] + eqns += src.inject(field=u.forward, expr=src * dt**2 / m, offset=model.nbpml) + op = Operator(eqns, subs=subs) + op.apply(u=u, m=m, damp=damp, src=src, t=1) + + def test_acoustic_w_src_w_rec(self, model, stencil, subs, m, damp, u, src, rec): + """ + Test that the acoustic wave equation forward operator produces the correct + results when running the same model as in ``test_adjointA.py``. + """ + dt = model.critical_dt + u.data[:] = 0.0 + eqns = [Eq(u.forward, stencil)] + eqns += src.inject(field=u.forward, expr=src * dt**2 / m, offset=model.nbpml) + eqns += rec.interpolate(expr=u, offset=model.nbpml) + op = Operator(eqns, subs=subs) + op.apply(u=u, m=m, damp=damp, src=src, rec=rec, t=1) + from IPython import embed; embed() From 7320df168de142cafe14bd147010cbc3057a03c0 Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Tue, 19 Sep 2017 16:48:33 +0200 Subject: [PATCH 06/11] flake8 fixes --- tests/test_yask.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/tests/test_yask.py b/tests/test_yask.py index 398b2c190c..cfe13975b8 100644 --- a/tests/test_yask.py +++ b/tests/test_yask.py @@ -6,7 +6,7 @@ pexpect = pytest.importorskip('yask_compiler') # Run only if YASK is available from devito import (Operator, DenseData, TimeData, PointData, - time, t, x, y, z, configuration, clear_cache, info) # noqa + time, t, x, y, z, configuration, clear_cache) # noqa from devito.dle import retrieve_iteration_tree # noqa from devito.yask.wrappers import YaskGrid, contexts # noqa @@ -311,7 +311,6 @@ class TestOperatorAcoustic(object): def model(self): shape = (60, 70, 80) nbpml = 10 - mkey = 'layers' return demo_model(spacing=[15, 15, 15], shape=shape, nbpml=nbpml, preset='layers', ratio=3) @@ -349,7 +348,6 @@ def stencil(self, m, damp, u): @pytest.fixture def src(self, model, time_params): - dt = model.critical_dt time_values = np.linspace(*time_params) # Discretized time axis # Define source geometry (center of domain, just below surface) src = RickerSource(name='src', ndim=model.dim, f0=0.01, time=time_values) @@ -407,4 +405,4 @@ def test_acoustic_w_src_w_rec(self, model, stencil, subs, m, damp, u, src, rec): eqns += rec.interpolate(expr=u, offset=model.nbpml) op = Operator(eqns, subs=subs) op.apply(u=u, m=m, damp=damp, src=src, rec=rec, t=1) - from IPython import embed; embed() + # TODO: come up with proper asserts, but u.data looks sane already From 444b1687cc88761d2e0a60d48840aa0bcff9d60e Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Thu, 21 Sep 2017 12:12:27 +0200 Subject: [PATCH 07/11] yask: Show performance after op.apply() --- devito/operator.py | 5 ++++- devito/yask/operator.py | 3 +++ 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/devito/operator.py b/devito/operator.py index cffcfe6ac3..f053635e01 100644 --- a/devito/operator.py +++ b/devito/operator.py @@ -453,13 +453,16 @@ def apply(self, **kwargs): self.cfunction(*list(arguments.values())) # Output summary of performance achieved + return self._profile_output(dim_sizes) + + def _profile_output(self, dim_sizes): + """Return a performance summary of the profiled sections.""" summary = self.profiler.summary(dim_sizes, self.dtype) with bar(): for k, v in summary.items(): name = '%s<%s>' % (k, ','.join('%d' % i for i in v.itershape)) info("Section %s with OI=%.2f computed in %.3f s [Perf: %.2f GFlops/s]" % (name, v.oi, v.time, v.gflopss)) - return summary def _profile_sections(self, nodes, parameters): diff --git a/devito/yask/operator.py b/devito/yask/operator.py index c5ec834841..8e95ea2d52 100644 --- a/devito/yask/operator.py +++ b/devito/yask/operator.py @@ -196,6 +196,9 @@ def apply(self, **kwargs): self.cfunction(*list(arguments.values())) log("YASK Operator successfully run!") + # Output summary of performance achieved + return self._profile_output(dim_sizes) + @property def compile(self): """ From 2a364ead23555631a7b65ffbc9ddbeb98459ccab Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Thu, 21 Sep 2017 12:30:37 +0200 Subject: [PATCH 08/11] yask: Add developer mode option --- devito/yask/__init__.py | 1 + devito/yask/wrappers.py | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/devito/yask/__init__.py b/devito/yask/__init__.py index a5712ca59a..4d0708db52 100644 --- a/devito/yask/__init__.py +++ b/devito/yask/__init__.py @@ -79,6 +79,7 @@ def __init__(self, *args, **kwargs): yask_configuration = Parameters('YASK-Configuration') yask_configuration.add('compiler', YaskCompiler()) yask_configuration.add('python-exec', False, [False, True]) +yask_configuration.add('develop-mode', True, [False, True]) # TODO: this should be somewhat sniffed yask_configuration.add('arch', 'snb', ['snb']) yask_configuration.add('isa', 'cpp', ['cpp']) diff --git a/devito/yask/wrappers.py b/devito/yask/wrappers.py index cee6400fdf..d4e6e53a6f 100644 --- a/devito/yask/wrappers.py +++ b/devito/yask/wrappers.py @@ -229,7 +229,8 @@ def __init__(self, name, yc_soln, domain): # JIT-compile it try: - make(os.environ['YASK_HOME'], ['-j', 'YK_CXXOPT=-O0', + opt_level = 1 if yask_configuration['develop-mode'] else 3 + make(os.environ['YASK_HOME'], ['-j', 'YK_CXXOPT=-O%d' % opt_level, # "EXTRA_MACROS=TRACE", 'YK_BASE=%s' % str(name), 'stencil=%s' % yc_soln.get_name(), From 56592094da9d469a007035cd81a929e6663b2e70 Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Fri, 22 Sep 2017 11:05:48 +0200 Subject: [PATCH 09/11] yask: Add proper asserts to the acoustic test --- tests/test_yask.py | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/tests/test_yask.py b/tests/test_yask.py index cfe13975b8..8a88fe0c3d 100644 --- a/tests/test_yask.py +++ b/tests/test_yask.py @@ -396,7 +396,7 @@ def test_acoustic_w_src_wo_rec(self, model, stencil, subs, m, damp, u, src): def test_acoustic_w_src_w_rec(self, model, stencil, subs, m, damp, u, src, rec): """ Test that the acoustic wave equation forward operator produces the correct - results when running the same model as in ``test_adjointA.py``. + results when running a 3D model also used in ``test_adjointA.py``. """ dt = model.critical_dt u.data[:] = 0.0 @@ -405,4 +405,12 @@ def test_acoustic_w_src_w_rec(self, model, stencil, subs, m, damp, u, src, rec): eqns += rec.interpolate(expr=u, offset=model.nbpml) op = Operator(eqns, subs=subs) op.apply(u=u, m=m, damp=damp, src=src, rec=rec, t=1) - # TODO: come up with proper asserts, but u.data looks sane already + + # TODO: the following "hacky" way of asserting correctness will be replaced + # once adjoint operators could be run through YASK. At the moment, the following + # expected norms have been "manually" derived from an analogous test (same + # equation, same model, ...) in test_adjointA.py + exp_u = 152.76 + exp_rec = 212.00 + assert np.isclose(np.linalg.norm(u.data[:]), exp_u, atol=exp_u*1.e-2) + assert np.isclose(np.linalg.norm(rec.data), exp_rec, atol=exp_rec*1.e-2) From 3266dd29f485993a1cdcbd25c3f4adb93ae7357b Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Fri, 22 Sep 2017 14:52:30 +0200 Subject: [PATCH 10/11] yask: Sniff ISA and ARCH on startup --- devito/yask/__init__.py | 19 ++++++++++++++++--- devito/yask/wrappers.py | 1 + 2 files changed, 17 insertions(+), 3 deletions(-) diff --git a/devito/yask/__init__.py b/devito/yask/__init__.py index 4d0708db52..1e9799ddde 100644 --- a/devito/yask/__init__.py +++ b/devito/yask/__init__.py @@ -6,6 +6,8 @@ from collections import OrderedDict import os +import cpuinfo + from devito import configuration from devito.exceptions import InvalidOperator from devito.logger import yask as log @@ -80,9 +82,20 @@ def __init__(self, *args, **kwargs): yask_configuration.add('compiler', YaskCompiler()) yask_configuration.add('python-exec', False, [False, True]) yask_configuration.add('develop-mode', True, [False, True]) -# TODO: this should be somewhat sniffed -yask_configuration.add('arch', 'snb', ['snb']) -yask_configuration.add('isa', 'cpp', ['cpp']) +# Sniff the highest Instruction Set Architecture level that we can YASK to use +isa, ISAs = 'cpp', ['cpp', 'avx', 'avx2', 'avx512', 'knc'] +if yask_configuration['develop-mode'] is False: + cpu_flags = cpuinfo.get_cpu_info()['flags'] + for i in reversed(ISAs): + if i in cpu_flags: + isa = i + break +yask_configuration.add('isa', isa, ISAs) +# Currently YASK also require the CPU architecture (e.g., snb for sandy bridge, +# hsw for haswell, etc.). At the moment, we simply infer it from the ISA +arch_mapper = {'cpp': 'intel64', 'avx': 'snb', 'avx2': 'hsw', 'avx512': 'knl'} +yask_configuration.add('arch', arch_mapper[isa], arch_mapper.values()) + configuration.add('yask', yask_configuration) log("Backend successfully initialized!") diff --git a/devito/yask/wrappers.py b/devito/yask/wrappers.py index d4e6e53a6f..9890b186c9 100644 --- a/devito/yask/wrappers.py +++ b/devito/yask/wrappers.py @@ -234,6 +234,7 @@ def __init__(self, name, yc_soln, domain): # "EXTRA_MACROS=TRACE", 'YK_BASE=%s' % str(name), 'stencil=%s' % yc_soln.get_name(), + 'arch=%s' % yask_configuration['arch'], '-C', namespace['kernel-path'], 'api']) except CompilationError: exit("Kernel solution compilation") From 3a2ae695838ae82c8d01c648880482fba02bf7fa Mon Sep 17 00:00:00 2001 From: Fabio Luporini Date: Fri, 22 Sep 2017 14:52:45 +0200 Subject: [PATCH 11/11] yask: Clean up lib directory before running tests --- tests/test_yask.py | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/tests/test_yask.py b/tests/test_yask.py index 8a88fe0c3d..7339557f0e 100644 --- a/tests/test_yask.py +++ b/tests/test_yask.py @@ -18,6 +18,13 @@ reason="'yask' wasn't selected as backend on startup") +def setup_module(module): + """Get rid of any YASK modules generated and JIT-compiled in previous runs. + This is not strictly necessary for the tests, but it helps in keeping the + lib directory clean, which may be helpful for offline analysis.""" + contexts.dump() + + @pytest.fixture(scope="module") def u(dims): u = DenseData(name='yu3D', shape=(16, 16, 16), dimensions=(x, y, z), space_order=0)