summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--.gitignore12
-rw-r--r--AUTHORS21
-rw-r--r--COPYING38
-rw-r--r--INSTALL59
-rw-r--r--NEWS146
-rw-r--r--PACKAGING49
-rw-r--r--README13
-rw-r--r--doc/layout.xml187
-rw-r--r--doc/reference.doxygen.in2415
-rw-r--r--doc/style.css691
-rw-r--r--src/cocoa_in_gtk2.mm433
-rw-r--r--src/cocoa_in_qt5.mm166
-rw-r--r--src/gtk2_in_qt4.cpp147
-rw-r--r--src/gtk2_in_qt5.cpp157
-rw-r--r--src/host.c90
-rw-r--r--src/instance.c390
-rw-r--r--src/qt4_in_gtk2.cpp154
-rw-r--r--src/qt5_in_gtk2.cpp237
-rw-r--r--src/suil_internal.h185
-rw-r--r--src/win_in_gtk2.cpp258
-rw-r--r--src/x11.c28
-rw-r--r--src/x11_in_gtk2.c398
-rw-r--r--src/x11_in_gtk3.c402
-rw-r--r--src/x11_in_qt4.cpp151
-rw-r--r--src/x11_in_qt5.cpp148
-rw-r--r--suil.pc.in10
-rw-r--r--suil/suil.h300
-rwxr-xr-xwaf171
-rw-r--r--waflib/.gitignore2
-rw-r--r--waflib/Build.py (renamed from Build.py)0
-rw-r--r--waflib/COPYING25
-rw-r--r--waflib/ConfigSet.py (renamed from ConfigSet.py)0
-rw-r--r--waflib/Configure.py (renamed from Configure.py)0
-rw-r--r--waflib/Context.py (renamed from Context.py)0
-rw-r--r--waflib/Errors.py (renamed from Errors.py)0
-rw-r--r--waflib/Logs.py (renamed from Logs.py)0
-rw-r--r--waflib/Node.py (renamed from Node.py)0
-rw-r--r--waflib/Options.py (renamed from Options.py)0
-rw-r--r--waflib/README.md (renamed from README.md)0
-rw-r--r--waflib/Runner.py (renamed from Runner.py)0
-rw-r--r--waflib/Scripting.py (renamed from Scripting.py)0
-rw-r--r--waflib/Task.py (renamed from Task.py)0
-rw-r--r--waflib/TaskGen.py (renamed from TaskGen.py)0
-rw-r--r--waflib/Tools/__init__.py (renamed from Tools/__init__.py)0
-rw-r--r--waflib/Tools/ar.py (renamed from Tools/ar.py)0
-rw-r--r--waflib/Tools/asm.py (renamed from Tools/asm.py)0
-rw-r--r--waflib/Tools/bison.py (renamed from Tools/bison.py)0
-rw-r--r--waflib/Tools/c.py (renamed from Tools/c.py)0
-rw-r--r--waflib/Tools/c_aliases.py (renamed from Tools/c_aliases.py)0
-rw-r--r--waflib/Tools/c_config.py (renamed from Tools/c_config.py)0
-rw-r--r--waflib/Tools/c_osx.py (renamed from Tools/c_osx.py)0
-rw-r--r--waflib/Tools/c_preproc.py (renamed from Tools/c_preproc.py)0
-rw-r--r--waflib/Tools/c_tests.py (renamed from Tools/c_tests.py)0
-rw-r--r--waflib/Tools/ccroot.py (renamed from Tools/ccroot.py)0
-rw-r--r--waflib/Tools/clang.py (renamed from Tools/clang.py)0
-rw-r--r--waflib/Tools/clangxx.py (renamed from Tools/clangxx.py)0
-rw-r--r--waflib/Tools/compiler_c.py (renamed from Tools/compiler_c.py)0
-rw-r--r--waflib/Tools/compiler_cxx.py (renamed from Tools/compiler_cxx.py)0
-rw-r--r--waflib/Tools/compiler_d.py (renamed from Tools/compiler_d.py)0
-rw-r--r--waflib/Tools/compiler_fc.py (renamed from Tools/compiler_fc.py)0
-rw-r--r--waflib/Tools/cs.py (renamed from Tools/cs.py)0
-rw-r--r--waflib/Tools/cxx.py (renamed from Tools/cxx.py)0
-rw-r--r--waflib/Tools/d.py (renamed from Tools/d.py)0
-rw-r--r--waflib/Tools/d_config.py (renamed from Tools/d_config.py)0
-rw-r--r--waflib/Tools/d_scan.py (renamed from Tools/d_scan.py)0
-rw-r--r--waflib/Tools/dbus.py (renamed from Tools/dbus.py)0
-rw-r--r--waflib/Tools/dmd.py (renamed from Tools/dmd.py)0
-rw-r--r--waflib/Tools/errcheck.py (renamed from Tools/errcheck.py)0
-rw-r--r--waflib/Tools/fc.py (renamed from Tools/fc.py)0
-rw-r--r--waflib/Tools/fc_config.py (renamed from Tools/fc_config.py)0
-rw-r--r--waflib/Tools/fc_scan.py (renamed from Tools/fc_scan.py)0
-rw-r--r--waflib/Tools/flex.py (renamed from Tools/flex.py)0
-rw-r--r--waflib/Tools/g95.py (renamed from Tools/g95.py)0
-rw-r--r--waflib/Tools/gas.py (renamed from Tools/gas.py)0
-rw-r--r--waflib/Tools/gcc.py (renamed from Tools/gcc.py)0
-rw-r--r--waflib/Tools/gdc.py (renamed from Tools/gdc.py)0
-rw-r--r--waflib/Tools/gfortran.py (renamed from Tools/gfortran.py)0
-rw-r--r--waflib/Tools/glib2.py (renamed from Tools/glib2.py)0
-rw-r--r--waflib/Tools/gnu_dirs.py (renamed from Tools/gnu_dirs.py)0
-rw-r--r--waflib/Tools/gxx.py (renamed from Tools/gxx.py)0
-rw-r--r--waflib/Tools/icc.py (renamed from Tools/icc.py)0
-rw-r--r--waflib/Tools/icpc.py (renamed from Tools/icpc.py)0
-rw-r--r--waflib/Tools/ifort.py (renamed from Tools/ifort.py)0
-rw-r--r--waflib/Tools/intltool.py (renamed from Tools/intltool.py)0
-rw-r--r--waflib/Tools/irixcc.py (renamed from Tools/irixcc.py)0
-rw-r--r--waflib/Tools/javaw.py (renamed from Tools/javaw.py)0
-rw-r--r--waflib/Tools/ldc2.py (renamed from Tools/ldc2.py)0
-rw-r--r--waflib/Tools/lua.py (renamed from Tools/lua.py)0
-rw-r--r--waflib/Tools/md5_tstamp.py (renamed from Tools/md5_tstamp.py)0
-rw-r--r--waflib/Tools/msvc.py (renamed from Tools/msvc.py)0
-rw-r--r--waflib/Tools/nasm.py (renamed from Tools/nasm.py)0
-rw-r--r--waflib/Tools/nobuild.py (renamed from Tools/nobuild.py)0
-rw-r--r--waflib/Tools/perl.py (renamed from Tools/perl.py)0
-rw-r--r--waflib/Tools/python.py (renamed from Tools/python.py)0
-rw-r--r--waflib/Tools/qt5.py (renamed from Tools/qt5.py)0
-rw-r--r--waflib/Tools/ruby.py (renamed from Tools/ruby.py)0
-rw-r--r--waflib/Tools/suncc.py (renamed from Tools/suncc.py)0
-rw-r--r--waflib/Tools/suncxx.py (renamed from Tools/suncxx.py)0
-rw-r--r--waflib/Tools/tex.py (renamed from Tools/tex.py)0
-rw-r--r--waflib/Tools/vala.py (renamed from Tools/vala.py)0
-rw-r--r--waflib/Tools/waf_unit_test.py (renamed from Tools/waf_unit_test.py)0
-rw-r--r--waflib/Tools/winres.py (renamed from Tools/winres.py)0
-rw-r--r--waflib/Tools/xlc.py (renamed from Tools/xlc.py)0
-rw-r--r--waflib/Tools/xlcxx.py (renamed from Tools/xlcxx.py)0
-rw-r--r--waflib/Utils.py (renamed from Utils.py)0
-rw-r--r--waflib/__init__.py (renamed from __init__.py)0
-rw-r--r--waflib/ansiterm.py (renamed from ansiterm.py)0
-rw-r--r--waflib/extras/__init__.py (renamed from extras/__init__.py)0
-rw-r--r--waflib/extras/autowaf.py (renamed from extras/autowaf.py)0
-rw-r--r--waflib/extras/batched_cc.py (renamed from extras/batched_cc.py)0
-rw-r--r--waflib/extras/biber.py (renamed from extras/biber.py)0
-rw-r--r--waflib/extras/bjam.py (renamed from extras/bjam.py)0
-rw-r--r--waflib/extras/blender.py (renamed from extras/blender.py)0
-rw-r--r--waflib/extras/boo.py (renamed from extras/boo.py)0
-rw-r--r--waflib/extras/boost.py (renamed from extras/boost.py)0
-rw-r--r--waflib/extras/build_file_tracker.py (renamed from extras/build_file_tracker.py)0
-rw-r--r--waflib/extras/build_logs.py (renamed from extras/build_logs.py)0
-rw-r--r--waflib/extras/buildcopy.py (renamed from extras/buildcopy.py)0
-rw-r--r--waflib/extras/c_bgxlc.py (renamed from extras/c_bgxlc.py)0
-rw-r--r--waflib/extras/c_dumbpreproc.py (renamed from extras/c_dumbpreproc.py)0
-rw-r--r--waflib/extras/c_emscripten.py (renamed from extras/c_emscripten.py)0
-rw-r--r--waflib/extras/c_nec.py (renamed from extras/c_nec.py)0
-rw-r--r--waflib/extras/cabal.py (renamed from extras/cabal.py)0
-rw-r--r--waflib/extras/cfg_altoptions.py (renamed from extras/cfg_altoptions.py)0
-rw-r--r--waflib/extras/clang_compilation_database.py (renamed from extras/clang_compilation_database.py)0
-rw-r--r--waflib/extras/codelite.py (renamed from extras/codelite.py)0
-rw-r--r--waflib/extras/color_gcc.py (renamed from extras/color_gcc.py)0
-rw-r--r--waflib/extras/color_rvct.py (renamed from extras/color_rvct.py)0
-rw-r--r--waflib/extras/compat15.py (renamed from extras/compat15.py)0
-rw-r--r--waflib/extras/cppcheck.py (renamed from extras/cppcheck.py)0
-rw-r--r--waflib/extras/cpplint.py (renamed from extras/cpplint.py)0
-rw-r--r--waflib/extras/cross_gnu.py (renamed from extras/cross_gnu.py)0
-rw-r--r--waflib/extras/cython.py (renamed from extras/cython.py)0
-rw-r--r--waflib/extras/dcc.py (renamed from extras/dcc.py)0
-rw-r--r--waflib/extras/distnet.py (renamed from extras/distnet.py)0
-rw-r--r--waflib/extras/doxygen.py (renamed from extras/doxygen.py)0
-rw-r--r--waflib/extras/dpapi.py (renamed from extras/dpapi.py)0
-rw-r--r--waflib/extras/eclipse.py (renamed from extras/eclipse.py)0
-rw-r--r--waflib/extras/erlang.py (renamed from extras/erlang.py)0
-rw-r--r--waflib/extras/fast_partial.py (renamed from extras/fast_partial.py)0
-rw-r--r--waflib/extras/fc_bgxlf.py (renamed from extras/fc_bgxlf.py)0
-rw-r--r--waflib/extras/fc_cray.py (renamed from extras/fc_cray.py)0
-rw-r--r--waflib/extras/fc_nag.py (renamed from extras/fc_nag.py)0
-rw-r--r--waflib/extras/fc_nec.py (renamed from extras/fc_nec.py)0
-rw-r--r--waflib/extras/fc_open64.py (renamed from extras/fc_open64.py)0
-rw-r--r--waflib/extras/fc_pgfortran.py (renamed from extras/fc_pgfortran.py)0
-rw-r--r--waflib/extras/fc_solstudio.py (renamed from extras/fc_solstudio.py)0
-rw-r--r--waflib/extras/fc_xlf.py (renamed from extras/fc_xlf.py)0
-rw-r--r--waflib/extras/file_to_object.py (renamed from extras/file_to_object.py)0
-rw-r--r--waflib/extras/fluid.py (renamed from extras/fluid.py)0
-rw-r--r--waflib/extras/freeimage.py (renamed from extras/freeimage.py)0
-rw-r--r--waflib/extras/fsb.py (renamed from extras/fsb.py)0
-rw-r--r--waflib/extras/fsc.py (renamed from extras/fsc.py)0
-rw-r--r--waflib/extras/gccdeps.py (renamed from extras/gccdeps.py)0
-rw-r--r--waflib/extras/gdbus.py (renamed from extras/gdbus.py)0
-rw-r--r--waflib/extras/gob2.py (renamed from extras/gob2.py)0
-rw-r--r--waflib/extras/halide.py (renamed from extras/halide.py)0
-rwxr-xr-xwaflib/extras/javatest.py (renamed from extras/javatest.py)0
-rw-r--r--waflib/extras/kde4.py (renamed from extras/kde4.py)0
-rw-r--r--waflib/extras/local_rpath.py (renamed from extras/local_rpath.py)0
-rw-r--r--waflib/extras/lv2.py (renamed from extras/lv2.py)0
-rw-r--r--waflib/extras/make.py (renamed from extras/make.py)0
-rw-r--r--waflib/extras/midl.py (renamed from extras/midl.py)0
-rw-r--r--waflib/extras/msvcdeps.py (renamed from extras/msvcdeps.py)0
-rw-r--r--waflib/extras/msvs.py (renamed from extras/msvs.py)0
-rw-r--r--waflib/extras/netcache_client.py (renamed from extras/netcache_client.py)0
-rw-r--r--waflib/extras/objcopy.py (renamed from extras/objcopy.py)0
-rw-r--r--waflib/extras/ocaml.py (renamed from extras/ocaml.py)0
-rw-r--r--waflib/extras/package.py (renamed from extras/package.py)0
-rw-r--r--waflib/extras/parallel_debug.py (renamed from extras/parallel_debug.py)0
-rw-r--r--waflib/extras/pch.py (renamed from extras/pch.py)0
-rw-r--r--waflib/extras/pep8.py (renamed from extras/pep8.py)0
-rw-r--r--waflib/extras/pgicc.py (renamed from extras/pgicc.py)0
-rw-r--r--waflib/extras/pgicxx.py (renamed from extras/pgicxx.py)0
-rw-r--r--waflib/extras/proc.py (renamed from extras/proc.py)0
-rw-r--r--waflib/extras/protoc.py (renamed from extras/protoc.py)0
-rw-r--r--waflib/extras/pyqt5.py (renamed from extras/pyqt5.py)0
-rw-r--r--waflib/extras/pytest.py (renamed from extras/pytest.py)0
-rw-r--r--waflib/extras/qnxnto.py (renamed from extras/qnxnto.py)0
-rw-r--r--waflib/extras/qt4.py (renamed from extras/qt4.py)0
-rw-r--r--waflib/extras/relocation.py (renamed from extras/relocation.py)0
-rw-r--r--waflib/extras/remote.py (renamed from extras/remote.py)0
-rw-r--r--waflib/extras/resx.py (renamed from extras/resx.py)0
-rw-r--r--waflib/extras/review.py (renamed from extras/review.py)0
-rw-r--r--waflib/extras/rst.py (renamed from extras/rst.py)0
-rw-r--r--waflib/extras/run_do_script.py (renamed from extras/run_do_script.py)0
-rw-r--r--waflib/extras/run_m_script.py (renamed from extras/run_m_script.py)0
-rw-r--r--waflib/extras/run_py_script.py (renamed from extras/run_py_script.py)0
-rw-r--r--waflib/extras/run_r_script.py (renamed from extras/run_r_script.py)0
-rw-r--r--waflib/extras/sas.py (renamed from extras/sas.py)0
-rw-r--r--waflib/extras/satellite_assembly.py (renamed from extras/satellite_assembly.py)0
-rw-r--r--waflib/extras/scala.py (renamed from extras/scala.py)0
-rw-r--r--waflib/extras/slow_qt4.py (renamed from extras/slow_qt4.py)0
-rw-r--r--waflib/extras/softlink_libs.py (renamed from extras/softlink_libs.py)0
-rw-r--r--waflib/extras/stale.py (renamed from extras/stale.py)0
-rw-r--r--waflib/extras/stracedeps.py (renamed from extras/stracedeps.py)0
-rw-r--r--waflib/extras/swig.py (renamed from extras/swig.py)0
-rw-r--r--waflib/extras/syms.py (renamed from extras/syms.py)0
-rw-r--r--waflib/extras/ticgt.py (renamed from extras/ticgt.py)0
-rw-r--r--waflib/extras/unity.py (renamed from extras/unity.py)0
-rw-r--r--waflib/extras/use_config.py (renamed from extras/use_config.py)0
-rw-r--r--waflib/extras/valadoc.py (renamed from extras/valadoc.py)0
-rw-r--r--waflib/extras/waf_xattr.py (renamed from extras/waf_xattr.py)0
-rw-r--r--waflib/extras/why.py (renamed from extras/why.py)0
-rw-r--r--waflib/extras/win32_opts.py (renamed from extras/win32_opts.py)0
-rw-r--r--waflib/extras/wix.py (renamed from extras/wix.py)0
-rw-r--r--waflib/extras/xcode6.py (renamed from extras/xcode6.py)0
-rw-r--r--waflib/fixpy2.py (renamed from fixpy2.py)0
-rwxr-xr-xwaflib/processor.py (renamed from processor.py)0
-rwxr-xr-xwaflib/waf16
-rw-r--r--wscript422
211 files changed, 7885 insertions, 36 deletions
diff --git a/.gitignore b/.gitignore
index 8d35cb3..6c800d0 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,2 +1,10 @@
-__pycache__
-*.pyc
+/.lock-waf*
+/.waf-[0-9]*/
+/build/
+
+.DS_Store
+.DS_Store?
+.Spotlight-V100
+.Trashes
+ehthumbs.db
+Thumbs.db
diff --git a/AUTHORS b/AUTHORS
new file mode 100644
index 0000000..8617acd
--- /dev/null
+++ b/AUTHORS
@@ -0,0 +1,21 @@
+David Robillard <d@drobilla.net>
+
+Contributors:
+
+ * Devin Anderson <surfacepatterns@gmail.com>
+ * Fix reparenting of Gtk UIs in Qt
+ * Peter Nelson <peter@fuzzle.org>
+ * Fix crash when a broken UI returns a NULL descriptor
+ * Filipe Lopes <falktx@gmail.com>
+ * Idle interface fixes for X11 in Qt4
+ * Qt5 in Gtk2 support
+ * Robin Gareus <robin@gareus.org>
+ * Support for resizing X11 UIs in Gtk
+ * Cocoa in Gtk wrapper
+ * Numerous Windows fixes
+ * Rui Nuno Capela
+ * Fixes for X11 in Qt4
+ * Qt5 wrappers
+ * Stefan Westerfeld
+ * Qt5 in Gtk2 support
+ * Initialization API
diff --git a/COPYING b/COPYING
index a4147d2..cc6a925 100644
--- a/COPYING
+++ b/COPYING
@@ -1,25 +1,13 @@
-Redistribution and use in source and binary forms, with or without
-modification, are permitted provided that the following conditions
-are met:
-
-1. Redistributions of source code must retain the above copyright
- notice, this list of conditions and the following disclaimer.
-
-2. Redistributions in binary form must reproduce the above copyright
- notice, this list of conditions and the following disclaimer in the
- documentation and/or other materials provided with the distribution.
-
-3. The name of the author may not be used to endorse or promote products
- derived from this software without specific prior written permission.
-
-THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
-IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
-WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
-DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
-INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
-(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
-SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
-HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
-STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
-IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
-POSSIBILITY OF SUCH DAMAGE.
+Copyright 2007-2016 David Robillard <http://drobilla.net>
+
+Permission to use, copy, modify, and/or distribute this software for any
+purpose with or without fee is hereby granted, provided that the above
+copyright notice and this permission notice appear in all copies.
+
+THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
diff --git a/INSTALL b/INSTALL
new file mode 100644
index 0000000..623cddd
--- /dev/null
+++ b/INSTALL
@@ -0,0 +1,59 @@
+Installation Instructions
+=========================
+
+Basic Installation
+------------------
+
+Building this software requires only Python. To install with default options:
+
+ ./waf configure
+ ./waf
+ ./waf install
+
+You may need to become root for the install stage, for example:
+
+ sudo ./waf install
+
+Configuration Options
+---------------------
+
+All supported options can be viewed using the command:
+
+ ./waf --help
+
+Most options only need to be passed during the configure stage, for example:
+
+ ./waf configure --prefix=/usr
+ ./waf
+ ./waf install
+
+Compiler Configuration
+----------------------
+
+Several standard environment variables can be used to control how compilers are
+invoked:
+
+ * CC: Path to C compiler
+ * CFLAGS: C compiler options
+ * CXX: Path to C++ compiler
+ * CXXFLAGS: C++ compiler options
+ * CPPFLAGS: C preprocessor options
+ * LINKFLAGS: Linker options
+
+Installation Directories
+------------------------
+
+The --prefix option (or the PREFIX environment variable) can be used to change
+the prefix which all files are installed under. There are also several options
+allowing for more fine-tuned control, see the --help output for details.
+
+Packaging
+---------
+
+Everything can be installed to a specific root directory by passing a --destdir
+option to the install stage (or setting the DESTDIR environment variable),
+which adds a prefix to all install paths. For example:
+
+ ./waf configure --prefix=/usr
+ ./waf
+ ./waf install --destdir=/tmp/package
diff --git a/NEWS b/NEWS
new file mode 100644
index 0000000..710772f
--- /dev/null
+++ b/NEWS
@@ -0,0 +1,146 @@
+suil (0.10.1) unstable;
+
+ * Add support for Cocoa in Qt5
+ * Fix resizing and add idle and update rate support for Qt5 in Gtk2
+
+ -- David Robillard <d@drobilla.net> Tue, 03 Oct 2017 22:11:49 +0200
+
+suil (0.10.0) stable;
+
+ * Add support for X11 in Gtk3
+ * Add support for Qt5 in Gtk2
+ * Add suil_init() to support early initialization and passing any necessary
+ information that may be needed in the future (thanks Stefan Westerfeld)
+ * Fix minor memory errors
+ * Fix building with X11 against custom LV2 install path (thanks Robin Gareus)
+
+ -- David Robillard <d@drobilla.net> Tue, 03 Oct 2017 22:11:49 +0200
+
+suil (0.8.4) stable;
+
+ * Configure based on compiler target OS for cross-compilation
+ * Add Cocoa in Gtk wrapper (patch from Robin Gareus)
+ * Various Windows fixes (patches from Robin Gareus)
+ * Center X11 UIs in Gtk (patch from Robin Gareus)
+ * Fix initial size of resizable X11 UIs in Gtk (patch from Robin Gareus)
+ * Bubble X11 key events up to Gtk parent (patch from Filipe Coelho)
+ * Add Gtk2 and X11 in Qt5 wrappers (patch from Rui Nuno Capela)
+ * Fix compilation with -Wl,--no-undefined
+ * Fix a few minor/unlikely memory errors
+ * Gracefully handle failure to open wrapper
+ * Only report suil_ui_supported() if necessary wrapper is compiled in
+ * Upgrade to waf 1.8.14
+
+ -- David Robillard <d@drobilla.net> Mon, 19 Sep 2016 22:47:44 -0400
+
+suil (0.8.2) stable;
+
+ * Fix embedding several Qt UIs in Gtk
+ * Add configure options to disable all Gtk or Qt support
+ * Upgrade to waf 1.7.16
+
+ -- David Robillard <d@drobilla.net> Fri, 08 Aug 2014 18:18:00 -0400
+
+suil (0.8.0) stable;
+
+ * Add suil_instance_get_handle (patch from Rui Nuno Capela)
+ * Fix compilation errors on some systems
+ * Upgrade to waf 1.7.14
+
+ -- David Robillard <d@drobilla.net> Sat, 04 Jan 2014 16:06:56 -0500
+
+suil (0.6.16) stable;
+
+ * Fix suil_instance_extension_data() for UIs with NULL extension_data
+ * Fix crashes and resizing for X11 in Qt (patch from Rui Nuno Capela)
+
+ -- David Robillard <d@drobilla.net> Mon, 16 Sep 2013 23:48:57 -0400
+
+suil (0.6.14) stable;
+
+ * Print system error message if module fails to load
+ * Lower dependency from Gtk 2.24 introduced in 0.6.12
+ * Add support for new LV2 idle interface
+ * Support resizing for X11 in Gtk (patch from Robin Gareus)
+ * Upgrade to waf 1.7.11
+
+ -- David Robillard <d@drobilla.net> Fri, 09 Aug 2013 00:16:48 -0400
+
+suil (0.6.12) stable;
+
+ * Fix key events for X11 in Gtk without using a troublesome event filter
+ * Fix crash when a broken UI returns a NULL descriptor
+ * Fix compilation on BSD
+
+ -- David Robillard <d@drobilla.net> Fri, 22 Feb 2013 13:03:48 -0500
+
+suil (0.6.10) stable;
+
+ * Downgrade to waf 1.7.5, previous version does not build modules due to
+ package check breakage in waf 1.7.6
+
+ -- David Robillard <d@drobilla.net> Sat, 22 Dec 2012 23:08:06 -0500
+
+suil (0.6.8) stable;
+
+ * Fix crash in x11_in_gtk2 when event_filter fires before widget is realized
+ * Use libgtk-x11-2.0.so.0 (with .0 suffix) by default which is available on
+ systems without the dev package
+ * Update to waf 1.7.8 and autowaf r90 (install docs to versioned directory)
+
+ -- David Robillard <d@drobilla.net> Sat, 22 Dec 2012 21:11:23 -0500
+
+suil (0.6.6) stable;
+
+ * Fix embedding Gtk in Qt as a child widget (support reparenting)
+ * Support for wrapping native Windows UIs in Gtk2
+ * Gracefully handle UIs with no port_event method
+ * Replace host provided features that match Suil implemented features, rather
+ than passing UIs duplicate features
+ * Disable timestamps in HTML documentation for reproducible build
+
+ -- David Robillard <d@drobilla.net> Wed, 14 Nov 2012 11:17:03 -0500
+
+suil (0.6.4) stable;
+
+ * Correctly handle resizing for Gtk2 in Qt4
+ * Improve documentation
+
+ -- David Robillard <d@drobilla.net> Mon, 09 Jul 2012 15:11:06 -0400
+
+suil (0.6.2) stable;
+
+ * Fix crashes when wrapper widget is destroyed by toolkit before
+ suil cleanup function is called
+ * Link Gtk wrappers with 'nodelete' to avoid Glib type errors
+ * Allow run-time configuration of module directory via environment variable
+ SUIL_MODULE_DIR
+
+ -- David Robillard <d@drobilla.net> Thu, 24 May 2012 23:18:18 -0400
+
+suil (0.6.0) stable;
+
+ * Use path variables in pkgconfig files
+ * Add support for embedding X11 UIs (ui:X11UI)
+ * Support new LV2 UI features automatically if provided by host
+
+ -- David Robillard <d@drobilla.net> Wed, 18 Apr 2012 19:32:43 -0400
+
+suil (0.4.4) stable;
+
+ * Fix embedding Gtk2 Qt4 UIs in Qt4 hosts that do not link to Gtk2
+
+ -- David Robillard <d@drobilla.net> Sat, 11 Jun 2011 11:20:11 -0400
+
+suil (0.4.2) stable;
+
+ * Fix compilation issues on some systems
+ * Fix build system Python 3 compatibility
+
+ -- David Robillard <d@drobilla.net> Wed, 25 May 2011 19:00:00 -0400
+
+suil (0.4.0) stable;
+
+ * Initial release
+
+ -- David Robillard <d@drobilla.net> Tue, 24 May 2011 23:00:00 -0400
diff --git a/PACKAGING b/PACKAGING
new file mode 100644
index 0000000..735a1c6
--- /dev/null
+++ b/PACKAGING
@@ -0,0 +1,49 @@
+These are generic guidelines, but please see below for important Suil specific
+information.
+
+This library is designed to allow parallel installation of different major
+versions. To facilitate this, the shared library name, include directory, and
+pkg-config file are suffixed with the major version number of the library.
+
+For example, if this library was named "foo" and at version 1.x.y:
+
+/usr/include/foo-1/foo/foo.h
+/usr/lib/foo-1.so.1.x.y
+/usr/lib/pkgconfig/foo-1.pc
+
+Dependencies check for pkg-config name "foo-1" and will build
+against a compatible version 1, regardless any other installed versions.
+
+*** IMPORTANT GUIDELINES FOR PACKAGERS ***
+
+Packages should follow the same conventions as above, i.e. include the major
+version (and only the major version) in the name of the package. Continuing the
+example above, the package(s) would be named foo-1 and foo-1-dev. This way,
+if/when version 2 comes out, it may be installed at the same time as version 1
+without breaking anything.
+
+Please do not create packages of this library that do not follow these
+guidelines, you will break things and cause unnecessary headaches. Please do
+not use any number as a suffix other than the actual major version number of the
+upstream source package.
+
+Because program and documentation names are not versioned, these should be
+included in separate packages which may replace previous versions, since
+there is little use in having parallel installations of them.
+
+*** IMPORTANT GUIDELINES FOR PACKAGING SUIL ***
+
+The purpose of Suil is to abstract plugin UI toolkits away from host code. To
+achieve this, Suil performs its magic by dynamically loading modules for each
+toolkit. The main Suil library does NOT depend on any toolkit libraries, and
+thus neither should your package. Please package the individual modules
+(e.g. libsuil_gtk2_in_qt4.so) as separate packages, which themselves depend on
+the involved toolkits. These packages should also be versioned as described
+above to support parallel installation.
+
+Please do not make the main Suil package depend on any toolkit package, this
+defeats the purpose of Suil and will severely irritate those who for whatever
+reason do not want a particular toolkit dependency. The main Suil package may
+have a weak dependency (e.g. "recommends") on the individual wrapper modules,
+and it's fine if these are installed by default, but it must be possible to
+install Suil without installing them if the user explicitly wishes to do so.
diff --git a/README b/README
new file mode 100644
index 0000000..bdfc668
--- /dev/null
+++ b/README
@@ -0,0 +1,13 @@
+Suil
+====
+
+Suil is a library for loading and wrapping LV2 plugin UIs.
+For more information, see <http://drobilla.net/software/suil>.
+
+With Suil, a host written in one supported toolkit can embed a plugin UI
+written in a different supported toolkit. Suil insulates hosts from toolkit
+libraries used by plugin UIs. For example, a Gtk host can embed a Qt UI
+without linking against Qt at compile time.
+
+ -- David Robillard <d@drobilla.net>
+
diff --git a/doc/layout.xml b/doc/layout.xml
new file mode 100644
index 0000000..74a109f
--- /dev/null
+++ b/doc/layout.xml
@@ -0,0 +1,187 @@
+<doxygenlayout version="1.0">
+ <!-- Navigation index tabs for HTML output -->
+ <navindex>
+ <tab type="mainpage" visible="yes" title=""/>
+ <tab type="pages" visible="yes" title="" intro=""/>
+ <tab type="modules" visible="yes" title="" intro=""/>
+ <tab type="namespaces" visible="yes" title="">
+ <tab type="namespacelist" visible="yes" title="" intro=""/>
+ <tab type="namespacemembers" visible="yes" title="" intro=""/>
+ </tab>
+ <tab type="classes" visible="yes" title="">
+ <tab type="classlist" visible="yes" title="" intro=""/>
+ <tab type="classindex" visible="$ALPHABETICAL_INDEX" title=""/>
+ <tab type="hierarchy" visible="yes" title="" intro=""/>
+ <tab type="classmembers" visible="yes" title="" intro=""/>
+ </tab>
+ <tab type="files" visible="yes" title="">
+ <tab type="filelist" visible="yes" title="" intro=""/>
+ <tab type="globals" visible="yes" title="" intro=""/>
+ </tab>
+ <tab type="examples" visible="yes" title="" intro=""/>
+ </navindex>
+
+ <!-- Layout definition for a class page -->
+ <class>
+ <briefdescription visible="yes"/>
+ <includes visible="$SHOW_INCLUDE_FILES"/>
+ <inheritancegraph visible="$CLASS_GRAPH"/>
+ <collaborationgraph visible="$COLLABORATION_GRAPH"/>
+ <allmemberslink visible="yes"/>
+ <memberdecl>
+ <nestedclasses visible="yes" title=""/>
+ <publictypes title=""/>
+ <publicslots title=""/>
+ <signals title=""/>
+ <publicmethods title=""/>
+ <publicstaticmethods title=""/>
+ <publicattributes title=""/>
+ <publicstaticattributes title=""/>
+ <protectedtypes title=""/>
+ <protectedslots title=""/>
+ <protectedmethods title=""/>
+ <protectedstaticmethods title=""/>
+ <protectedattributes title=""/>
+ <protectedstaticattributes title=""/>
+ <packagetypes title=""/>
+ <packagemethods title=""/>
+ <packagestaticmethods title=""/>
+ <packageattributes title=""/>
+ <packagestaticattributes title=""/>
+ <properties title=""/>
+ <events title=""/>
+ <privatetypes title=""/>
+ <privateslots title=""/>
+ <privatemethods title=""/>
+ <privatestaticmethods title=""/>
+ <privateattributes title=""/>
+ <privatestaticattributes title=""/>
+ <friends title=""/>
+ <related title="" subtitle=""/>
+ <membergroups visible="yes"/>
+ </memberdecl>
+ <detaileddescription title=""/>
+ <memberdef>
+ <inlineclasses title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <constructors title=""/>
+ <functions title=""/>
+ <related title=""/>
+ <variables title=""/>
+ <properties title=""/>
+ <events title=""/>
+ </memberdef>
+ <usedfiles visible="$SHOW_USED_FILES"/>
+ <authorsection visible="yes"/>
+ </class>
+
+ <!-- Layout definition for a namespace page -->
+ <namespace>
+ <briefdescription visible="yes"/>
+ <memberdecl>
+ <nestednamespaces visible="yes" title=""/>
+ <classes visible="yes" title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <functions title=""/>
+ <variables title=""/>
+ <membergroups visible="yes"/>
+ </memberdecl>
+ <detaileddescription title=""/>
+ <memberdef>
+ <inlineclasses title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <functions title=""/>
+ <variables title=""/>
+ </memberdef>
+ <authorsection visible="yes"/>
+ </namespace>
+
+ <!-- Layout definition for a file page -->
+ <file>
+ <briefdescription visible="yes"/>
+ <includes visible="$SHOW_INCLUDE_FILES"/>
+ <includegraph visible="$INCLUDE_GRAPH"/>
+ <includedbygraph visible="$INCLUDED_BY_GRAPH"/>
+ <sourcelink visible="yes"/>
+ <memberdecl>
+ <classes visible="yes" title=""/>
+ <namespaces visible="yes" title=""/>
+ <defines title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <functions title=""/>
+ <variables title=""/>
+ <membergroups visible="yes"/>
+ </memberdecl>
+ <detaileddescription title=""/>
+ <memberdef>
+ <inlineclasses title=""/>
+ <defines title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <functions title=""/>
+ <variables title=""/>
+ </memberdef>
+ <authorsection/>
+ </file>
+
+ <!-- Layout definition for a group page -->
+ <group>
+ <briefdescription visible="no"/>
+ <groupgraph visible="$GROUP_GRAPHS"/>
+ <detaileddescription title=""/>
+ <memberdecl>
+ <nestedgroups visible="yes" title=""/>
+ <dirs visible="yes" title=""/>
+ <files visible="yes" title=""/>
+ <namespaces visible="yes" title=""/>
+ <classes visible="yes" title=""/>
+ <defines title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <enumvalues title=""/>
+ <functions title=""/>
+ <variables title=""/>
+ <signals title=""/>
+ <publicslots title=""/>
+ <protectedslots title=""/>
+ <privateslots title=""/>
+ <events title=""/>
+ <properties title=""/>
+ <friends title=""/>
+ <membergroups visible="yes"/>
+ </memberdecl>
+ <memberdef>
+ <pagedocs/>
+ <inlineclasses title=""/>
+ <defines title=""/>
+ <typedefs title=""/>
+ <enums title=""/>
+ <enumvalues title=""/>
+ <functions title=""/>
+ <variables title=""/>
+ <signals title=""/>
+ <publicslots title=""/>
+ <protectedslots title=""/>
+ <privateslots title=""/>
+ <events title=""/>
+ <properties title=""/>
+ <friends title=""/>
+ </memberdef>
+ <authorsection visible="yes"/>
+ </group>
+
+ <!-- Layout definition for a directory page -->
+ <directory>
+ <briefdescription visible="yes"/>
+ <directorygraph visible="yes"/>
+ <memberdecl>
+ <dirs visible="yes"/>
+ <files visible="yes"/>
+ </memberdecl>
+ <detaileddescription title=""/>
+ </directory>
+</doxygenlayout>
diff --git a/doc/reference.doxygen.in b/doc/reference.doxygen.in
new file mode 100644
index 0000000..4ea36a3
--- /dev/null
+++ b/doc/reference.doxygen.in
@@ -0,0 +1,2415 @@
+# Doxyfile 1.8.12
+
+# This file describes the settings to be used by the documentation system
+# doxygen (www.doxygen.org) for a project.
+#
+# All text after a double hash (##) is considered a comment and is placed in
+# front of the TAG it is preceding.
+#
+# All text after a single hash (#) is considered a comment and will be ignored.
+# The format is:
+# TAG = value [value, ...]
+# For lists, items can also be appended using:
+# TAG += value [value, ...]
+# Values that contain spaces should be placed between quotes (\" \").
+
+#---------------------------------------------------------------------------
+# Project related configuration options
+#---------------------------------------------------------------------------
+
+# This tag specifies the encoding used for all characters in the config file
+# that follow. The default is UTF-8 which is also the encoding used for all text
+# before the first occurrence of this tag. Doxygen uses libiconv (or the iconv
+# built into libc) for the transcoding. See http://www.gnu.org/software/libiconv
+# for the list of possible encodings.
+# The default value is: UTF-8.
+
+DOXYFILE_ENCODING = UTF-8
+
+# The PROJECT_NAME tag is a single word (or a sequence of words surrounded by
+# double-quotes, unless you are using Doxywizard) that should identify the
+# project for which the documentation is generated. This name is used in the
+# title of most generated pages and in a few other places.
+# The default value is: My Project.
+
+PROJECT_NAME = Suil
+
+# The PROJECT_NUMBER tag can be used to enter a project or revision number. This
+# could be handy for archiving the generated documentation or if some version
+# control system is used.
+
+PROJECT_NUMBER = @SUIL_VERSION@
+
+# Using the PROJECT_BRIEF tag one can provide an optional one line description
+# for a project that appears at the top of each page and should give viewer a
+# quick idea about the purpose of the project. Keep the description short.
+
+PROJECT_BRIEF =
+
+# With the PROJECT_LOGO tag one can specify a logo or an icon that is included
+# in the documentation. The maximum height of the logo should not exceed 55
+# pixels and the maximum width should not exceed 200 pixels. Doxygen will copy
+# the logo to the output directory.
+
+PROJECT_LOGO =
+
+# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute) path
+# into which the generated documentation will be written. If a relative path is
+# entered, it will be relative to the location where doxygen was started. If
+# left blank the current directory will be used.
+
+OUTPUT_DIRECTORY = .
+
+# If the CREATE_SUBDIRS tag is set to YES then doxygen will create 4096 sub-
+# directories (in 2 levels) under the output directory of each output format and
+# will distribute the generated files over these directories. Enabling this
+# option can be useful when feeding doxygen a huge amount of source files, where
+# putting all generated files in the same directory would otherwise causes
+# performance problems for the file system.
+# The default value is: NO.
+
+CREATE_SUBDIRS = NO
+
+# If the ALLOW_UNICODE_NAMES tag is set to YES, doxygen will allow non-ASCII
+# characters to appear in the names of generated files. If set to NO, non-ASCII
+# characters will be escaped, for example _xE3_x81_x84 will be used for Unicode
+# U+3044.
+# The default value is: NO.
+
+ALLOW_UNICODE_NAMES = NO
+
+# The OUTPUT_LANGUAGE tag is used to specify the language in which all
+# documentation generated by doxygen is written. Doxygen will use this
+# information to generate all constant output in the proper language.
+# Possible values are: Afrikaans, Arabic, Armenian, Brazilian, Catalan, Chinese,
+# Chinese-Traditional, Croatian, Czech, Danish, Dutch, English (United States),
+# Esperanto, Farsi (Persian), Finnish, French, German, Greek, Hungarian,
+# Indonesian, Italian, Japanese, Japanese-en (Japanese with English messages),
+# Korean, Korean-en (Korean with English messages), Latvian, Lithuanian,
+# Macedonian, Norwegian, Persian (Farsi), Polish, Portuguese, Romanian, Russian,
+# Serbian, Serbian-Cyrillic, Slovak, Slovene, Spanish, Swedish, Turkish,
+# Ukrainian and Vietnamese.
+# The default value is: English.
+
+OUTPUT_LANGUAGE = English
+
+# If the BRIEF_MEMBER_DESC tag is set to YES, doxygen will include brief member
+# descriptions after the members that are listed in the file and class
+# documentation (similar to Javadoc). Set to NO to disable this.
+# The default value is: YES.
+
+BRIEF_MEMBER_DESC = NO
+
+# If the REPEAT_BRIEF tag is set to YES, doxygen will prepend the brief
+# description of a member or function before the detailed description
+#
+# Note: If both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the
+# brief descriptions will be completely suppressed.
+# The default value is: YES.
+
+REPEAT_BRIEF = YES
+
+# This tag implements a quasi-intelligent brief description abbreviator that is
+# used to form the text in various listings. Each string in this list, if found
+# as the leading text of the brief description, will be stripped from the text
+# and the result, after processing the whole list, is used as the annotated
+# text. Otherwise, the brief description is used as-is. If left blank, the
+# following values are used ($name is automatically replaced with the name of
+# the entity):The $name class, The $name widget, The $name file, is, provides,
+# specifies, contains, represents, a, an and the.
+
+ABBREVIATE_BRIEF =
+
+# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then
+# doxygen will generate a detailed section even if there is only a brief
+# description.
+# The default value is: NO.
+
+ALWAYS_DETAILED_SEC = NO
+
+# If the INLINE_INHERITED_MEMB tag is set to YES, doxygen will show all
+# inherited members of a class in the documentation of that class as if those
+# members were ordinary class members. Constructors, destructors and assignment
+# operators of the base classes will not be shown.
+# The default value is: NO.
+
+INLINE_INHERITED_MEMB = NO
+
+# If the FULL_PATH_NAMES tag is set to YES, doxygen will prepend the full path
+# before files name in the file list and in the header files. If set to NO the
+# shortest path that makes the file name unique will be used
+# The default value is: YES.
+
+FULL_PATH_NAMES = NO
+
+# The STRIP_FROM_PATH tag can be used to strip a user-defined part of the path.
+# Stripping is only done if one of the specified strings matches the left-hand
+# part of the path. The tag can be used to show relative paths in the file list.
+# If left blank the directory from which doxygen is run is used as the path to
+# strip.
+#
+# Note that you can specify absolute paths here, but also relative paths, which
+# will be relative from the directory where doxygen is started.
+# This tag requires that the tag FULL_PATH_NAMES is set to YES.
+
+STRIP_FROM_PATH =
+
+# The STRIP_FROM_INC_PATH tag can be used to strip a user-defined part of the
+# path mentioned in the documentation of a class, which tells the reader which
+# header file to include in order to use a class. If left blank only the name of
+# the header file containing the class definition is used. Otherwise one should
+# specify the list of include paths that are normally passed to the compiler
+# using the -I flag.
+
+STRIP_FROM_INC_PATH =
+
+# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter (but
+# less readable) file names. This can be useful is your file systems doesn't
+# support long names like on DOS, Mac, or CD-ROM.
+# The default value is: NO.
+
+SHORT_NAMES = NO
+
+# If the JAVADOC_AUTOBRIEF tag is set to YES then doxygen will interpret the
+# first line (until the first dot) of a Javadoc-style comment as the brief
+# description. If set to NO, the Javadoc-style will behave just like regular Qt-
+# style comments (thus requiring an explicit @brief command for a brief
+# description.)
+# The default value is: NO.
+
+JAVADOC_AUTOBRIEF = YES
+
+# If the QT_AUTOBRIEF tag is set to YES then doxygen will interpret the first
+# line (until the first dot) of a Qt-style comment as the brief description. If
+# set to NO, the Qt-style will behave just like regular Qt-style comments (thus
+# requiring an explicit \brief command for a brief description.)
+# The default value is: NO.
+
+QT_AUTOBRIEF = NO
+
+# The MULTILINE_CPP_IS_BRIEF tag can be set to YES to make doxygen treat a
+# multi-line C++ special comment block (i.e. a block of //! or /// comments) as
+# a brief description. This used to be the default behavior. The new default is
+# to treat a multi-line C++ comment block as a detailed description. Set this
+# tag to YES if you prefer the old behavior instead.
+#
+# Note that setting this tag to YES also means that rational rose comments are
+# not recognized any more.
+# The default value is: NO.
+
+MULTILINE_CPP_IS_BRIEF = NO
+
+# If the INHERIT_DOCS tag is set to YES then an undocumented member inherits the
+# documentation from any documented member that it re-implements.
+# The default value is: YES.
+
+INHERIT_DOCS = YES
+
+# If the SEPARATE_MEMBER_PAGES tag is set to YES then doxygen will produce a new
+# page for each member. If set to NO, the documentation of a member will be part
+# of the file/class/namespace that contains it.
+# The default value is: NO.
+
+SEPARATE_MEMBER_PAGES = NO
+
+# The TAB_SIZE tag can be used to set the number of spaces in a tab. Doxygen
+# uses this value to replace tabs by spaces in code fragments.
+# Minimum value: 1, maximum value: 16, default value: 4.
+
+TAB_SIZE = 4
+
+# This tag can be used to specify a number of aliases that act as commands in
+# the documentation. An alias has the form:
+# name=value
+# For example adding
+# "sideeffect=@par Side Effects:\n"
+# will allow you to put the command \sideeffect (or @sideeffect) in the
+# documentation, which will result in a user-defined paragraph with heading
+# "Side Effects:". You can put \n's in the value part of an alias to insert
+# newlines.
+
+ALIASES =
+
+# This tag can be used to specify a number of word-keyword mappings (TCL only).
+# A mapping has the form "name=value". For example adding "class=itcl::class"
+# will allow you to use the command class in the itcl::class meaning.
+
+TCL_SUBST =
+
+# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources
+# only. Doxygen will then generate output that is more tailored for C. For
+# instance, some of the names that are used will be different. The list of all
+# members will be omitted, etc.
+# The default value is: NO.
+
+OPTIMIZE_OUTPUT_FOR_C = YES
+
+# Set the OPTIMIZE_OUTPUT_JAVA tag to YES if your project consists of Java or
+# Python sources only. Doxygen will then generate output that is more tailored
+# for that language. For instance, namespaces will be presented as packages,
+# qualified scopes will look different, etc.
+# The default value is: NO.
+
+OPTIMIZE_OUTPUT_JAVA = NO
+
+# Set the OPTIMIZE_FOR_FORTRAN tag to YES if your project consists of Fortran
+# sources. Doxygen will then generate output that is tailored for Fortran.
+# The default value is: NO.
+
+OPTIMIZE_FOR_FORTRAN = NO
+
+# Set the OPTIMIZE_OUTPUT_VHDL tag to YES if your project consists of VHDL
+# sources. Doxygen will then generate output that is tailored for VHDL.
+# The default value is: NO.
+
+OPTIMIZE_OUTPUT_VHDL = NO
+
+# Doxygen selects the parser to use depending on the extension of the files it
+# parses. With this tag you can assign which parser to use for a given
+# extension. Doxygen has a built-in mapping, but you can override or extend it
+# using this tag. The format is ext=language, where ext is a file extension, and
+# language is one of the parsers supported by doxygen: IDL, Java, Javascript,
+# C#, C, C++, D, PHP, Objective-C, Python, Fortran (fixed format Fortran:
+# FortranFixed, free formatted Fortran: FortranFree, unknown formatted Fortran:
+# Fortran. In the later case the parser tries to guess whether the code is fixed
+# or free formatted code, this is the default for Fortran type files), VHDL. For
+# instance to make doxygen treat .inc files as Fortran files (default is PHP),
+# and .f files as C (default is Fortran), use: inc=Fortran f=C.
+#
+# Note: For files without extension you can use no_extension as a placeholder.
+#
+# Note that for custom extensions you also need to set FILE_PATTERNS otherwise
+# the files are not read by doxygen.
+
+EXTENSION_MAPPING =
+
+# If the MARKDOWN_SUPPORT tag is enabled then doxygen pre-processes all comments
+# according to the Markdown format, which allows for more readable
+# documentation. See http://daringfireball.net/projects/markdown/ for details.
+# The output of markdown processing is further processed by doxygen, so you can
+# mix doxygen, HTML, and XML commands with Markdown formatting. Disable only in
+# case of backward compatibilities issues.
+# The default value is: YES.
+
+MARKDOWN_SUPPORT = YES
+
+# When the TOC_INCLUDE_HEADINGS tag is set to a non-zero value, all headings up
+# to that level are automatically included in the table of contents, even if
+# they do not have an id attribute.
+# Note: This feature currently applies only to Markdown headings.
+# Minimum value: 0, maximum value: 99, default value: 0.
+# This tag requires that the tag MARKDOWN_SUPPORT is set to YES.
+
+TOC_INCLUDE_HEADINGS = 0
+
+# When enabled doxygen tries to link words that correspond to documented
+# classes, or namespaces to their corresponding documentation. Such a link can
+# be prevented in individual cases by putting a % sign in front of the word or
+# globally by setting AUTOLINK_SUPPORT to NO.
+# The default value is: YES.
+
+AUTOLINK_SUPPORT = YES
+
+# If you use STL classes (i.e. std::string, std::vector, etc.) but do not want
+# to include (a tag file for) the STL sources as input, then you should set this
+# tag to YES in order to let doxygen match functions declarations and
+# definitions whose arguments contain STL classes (e.g. func(std::string);
+# versus func(std::string) {}). This also make the inheritance and collaboration
+# diagrams that involve STL classes more complete and accurate.
+# The default value is: NO.
+
+BUILTIN_STL_SUPPORT = NO
+
+# If you use Microsoft's C++/CLI language, you should set this option to YES to
+# enable parsing support.
+# The default value is: NO.
+
+CPP_CLI_SUPPORT = NO
+
+# Set the SIP_SUPPORT tag to YES if your project consists of sip (see:
+# http://www.riverbankcomputing.co.uk/software/sip/intro) sources only. Doxygen
+# will parse them like normal C++ but will assume all classes use public instead
+# of private inheritance when no explicit protection keyword is present.
+# The default value is: NO.
+
+SIP_SUPPORT = NO
+
+# For Microsoft's IDL there are propget and propput attributes to indicate
+# getter and setter methods for a property. Setting this option to YES will make
+# doxygen to replace the get and set methods by a property in the documentation.
+# This will only work if the methods are indeed getting or setting a simple
+# type. If this is not the case, or you want to show the methods anyway, you
+# should set this option to NO.
+# The default value is: YES.
+
+IDL_PROPERTY_SUPPORT = YES
+
+# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC
+# tag is set to YES then doxygen will reuse the documentation of the first
+# member in the group (if any) for the other members of the group. By default
+# all members of a group must be documented explicitly.
+# The default value is: NO.
+
+DISTRIBUTE_GROUP_DOC = NO
+
+# If one adds a struct or class to a group and this option is enabled, then also
+# any nested class or struct is added to the same group. By default this option
+# is disabled and one has to add nested compounds explicitly via \ingroup.
+# The default value is: NO.
+
+GROUP_NESTED_COMPOUNDS = NO
+
+# Set the SUBGROUPING tag to YES to allow class member groups of the same type
+# (for instance a group of public functions) to be put as a subgroup of that
+# type (e.g. under the Public Functions section). Set it to NO to prevent
+# subgrouping. Alternatively, this can be done per class using the
+# \nosubgrouping command.
+# The default value is: YES.
+
+SUBGROUPING = YES
+
+# When the INLINE_GROUPED_CLASSES tag is set to YES, classes, structs and unions
+# are shown inside the group in which they are included (e.g. using \ingroup)
+# instead of on a separate page (for HTML and Man pages) or section (for LaTeX
+# and RTF).
+#
+# Note that this feature does not work in combination with
+# SEPARATE_MEMBER_PAGES.
+# The default value is: NO.
+
+INLINE_GROUPED_CLASSES = YES
+
+# When the INLINE_SIMPLE_STRUCTS tag is set to YES, structs, classes, and unions
+# with only public data fields or simple typedef fields will be shown inline in
+# the documentation of the scope in which they are defined (i.e. file,
+# namespace, or group documentation), provided this scope is documented. If set
+# to NO, structs, classes, and unions are shown on a separate page (for HTML and
+# Man pages) or section (for LaTeX and RTF).
+# The default value is: NO.
+
+INLINE_SIMPLE_STRUCTS = YES
+
+# When TYPEDEF_HIDES_STRUCT tag is enabled, a typedef of a struct, union, or
+# enum is documented as struct, union, or enum with the name of the typedef. So
+# typedef struct TypeS {} TypeT, will appear in the documentation as a struct
+# with name TypeT. When disabled the typedef will appear as a member of a file,
+# namespace, or class. And the struct will be named TypeS. This can typically be
+# useful for C code in case the coding convention dictates that all compound
+# types are typedef'ed and only the typedef is referenced, never the tag name.
+# The default value is: NO.
+
+TYPEDEF_HIDES_STRUCT = YES
+
+# The size of the symbol lookup cache can be set using LOOKUP_CACHE_SIZE. This
+# cache is used to resolve symbols given their name and scope. Since this can be
+# an expensive process and often the same symbol appears multiple times in the
+# code, doxygen keeps a cache of pre-resolved symbols. If the cache is too small
+# doxygen will become slower. If the cache is too large, memory is wasted. The
+# cache size is given by this formula: 2^(16+LOOKUP_CACHE_SIZE). The valid range
+# is 0..9, the default is 0, corresponding to a cache size of 2^16=65536
+# symbols. At the end of a run doxygen will report the cache usage and suggest
+# the optimal cache size from a speed point of view.
+# Minimum value: 0, maximum value: 9, default value: 0.
+
+LOOKUP_CACHE_SIZE = 0
+
+#---------------------------------------------------------------------------
+# Build related configuration options
+#---------------------------------------------------------------------------
+
+# If the EXTRACT_ALL tag is set to YES, doxygen will assume all entities in
+# documentation are documented, even if no documentation was available. Private
+# class members and static file members will be hidden unless the
+# EXTRACT_PRIVATE respectively EXTRACT_STATIC tags are set to YES.
+# Note: This will also disable the warnings about undocumented members that are
+# normally produced when WARNINGS is set to YES.
+# The default value is: NO.
+
+EXTRACT_ALL = YES
+
+# If the EXTRACT_PRIVATE tag is set to YES, all private members of a class will
+# be included in the documentation.
+# The default value is: NO.
+
+EXTRACT_PRIVATE = NO
+
+# If the EXTRACT_PACKAGE tag is set to YES, all members with package or internal
+# scope will be included in the documentation.
+# The default value is: NO.
+
+EXTRACT_PACKAGE = NO
+
+# If the EXTRACT_STATIC tag is set to YES, all static members of a file will be
+# included in the documentation.
+# The default value is: NO.
+
+EXTRACT_STATIC = YES
+
+# If the EXTRACT_LOCAL_CLASSES tag is set to YES, classes (and structs) defined
+# locally in source files will be included in the documentation. If set to NO,
+# only classes defined in header files are included. Does not have any effect
+# for Java sources.
+# The default value is: YES.
+
+EXTRACT_LOCAL_CLASSES = NO
+
+# This flag is only useful for Objective-C code. If set to YES, local methods,
+# which are defined in the implementation section but not in the interface are
+# included in the documentation. If set to NO, only methods in the interface are
+# included.
+# The default value is: NO.
+
+EXTRACT_LOCAL_METHODS = NO
+
+# If this flag is set to YES, the members of anonymous namespaces will be
+# extracted and appear in the documentation as a namespace called
+# 'anonymous_namespace{file}', where file will be replaced with the base name of
+# the file that contains the anonymous namespace. By default anonymous namespace
+# are hidden.
+# The default value is: NO.
+
+EXTRACT_ANON_NSPACES = NO
+
+# If the HIDE_UNDOC_MEMBERS tag is set to YES, doxygen will hide all
+# undocumented members inside documented classes or files. If set to NO these
+# members will be included in the various overviews, but no documentation
+# section is generated. This option has no effect if EXTRACT_ALL is enabled.
+# The default value is: NO.
+
+HIDE_UNDOC_MEMBERS = YES
+
+# If the HIDE_UNDOC_CLASSES tag is set to YES, doxygen will hide all
+# undocumented classes that are normally visible in the class hierarchy. If set
+# to NO, these classes will be included in the various overviews. This option
+# has no effect if EXTRACT_ALL is enabled.
+# The default value is: NO.
+
+HIDE_UNDOC_CLASSES = YES
+
+# If the HIDE_FRIEND_COMPOUNDS tag is set to YES, doxygen will hide all friend
+# (class|struct|union) declarations. If set to NO, these declarations will be
+# included in the documentation.
+# The default value is: NO.
+
+HIDE_FRIEND_COMPOUNDS = NO
+
+# If the HIDE_IN_BODY_DOCS tag is set to YES, doxygen will hide any
+# documentation blocks found inside the body of a function. If set to NO, these
+# blocks will be appended to the function's detailed documentation block.
+# The default value is: NO.
+
+HIDE_IN_BODY_DOCS = YES
+
+# The INTERNAL_DOCS tag determines if documentation that is typed after a
+# \internal command is included. If the tag is set to NO then the documentation
+# will be excluded. Set it to YES to include the internal documentation.
+# The default value is: NO.
+
+INTERNAL_DOCS = NO
+
+# If the CASE_SENSE_NAMES tag is set to NO then doxygen will only generate file
+# names in lower-case letters. If set to YES, upper-case letters are also
+# allowed. This is useful if you have classes or files whose names only differ
+# in case and if your file system supports case sensitive file names. Windows
+# and Mac users are advised to set this option to NO.
+# The default value is: system dependent.
+
+CASE_SENSE_NAMES = YES
+
+# If the HIDE_SCOPE_NAMES tag is set to NO then doxygen will show members with
+# their full class and namespace scopes in the documentation. If set to YES, the
+# scope will be hidden.
+# The default value is: NO.
+
+HIDE_SCOPE_NAMES = NO
+
+# If the HIDE_COMPOUND_REFERENCE tag is set to NO (default) then doxygen will
+# append additional text to a page's title, such as Class Reference. If set to
+# YES the compound reference will be hidden.
+# The default value is: NO.
+
+HIDE_COMPOUND_REFERENCE= NO
+
+# If the SHOW_INCLUDE_FILES tag is set to YES then doxygen will put a list of
+# the files that are included by a file in the documentation of that file.
+# The default value is: YES.
+
+SHOW_INCLUDE_FILES = YES
+
+# If the SHOW_GROUPED_MEMB_INC tag is set to YES then Doxygen will add for each
+# grouped member an include statement to the documentation, telling the reader
+# which file to include in order to use the member.
+# The default value is: NO.
+
+SHOW_GROUPED_MEMB_INC = NO
+
+# If the FORCE_LOCAL_INCLUDES tag is set to YES then doxygen will list include
+# files with double quotes in the documentation rather than with sharp brackets.
+# The default value is: NO.
+
+FORCE_LOCAL_INCLUDES = NO
+
+# If the INLINE_INFO tag is set to YES then a tag [inline] is inserted in the
+# documentation for inline members.
+# The default value is: YES.
+
+INLINE_INFO = YES
+
+# If the SORT_MEMBER_DOCS tag is set to YES then doxygen will sort the
+# (detailed) documentation of file and class members alphabetically by member
+# name. If set to NO, the members will appear in declaration order.
+# The default value is: YES.
+
+SORT_MEMBER_DOCS = NO
+
+# If the SORT_BRIEF_DOCS tag is set to YES then doxygen will sort the brief
+# descriptions of file, namespace and class members alphabetically by member
+# name. If set to NO, the members will appear in declaration order. Note that
+# this will also influence the order of the classes in the class list.
+# The default value is: NO.
+
+SORT_BRIEF_DOCS = NO
+
+# If the SORT_MEMBERS_CTORS_1ST tag is set to YES then doxygen will sort the
+# (brief and detailed) documentation of class members so that constructors and
+# destructors are listed first. If set to NO the constructors will appear in the
+# respective orders defined by SORT_BRIEF_DOCS and SORT_MEMBER_DOCS.
+# Note: If SORT_BRIEF_DOCS is set to NO this option is ignored for sorting brief
+# member documentation.
+# Note: If SORT_MEMBER_DOCS is set to NO this option is ignored for sorting
+# detailed member documentation.
+# The default value is: NO.
+
+SORT_MEMBERS_CTORS_1ST = NO
+
+# If the SORT_GROUP_NAMES tag is set to YES then doxygen will sort the hierarchy
+# of group names into alphabetical order. If set to NO the group names will
+# appear in their defined order.
+# The default value is: NO.
+
+SORT_GROUP_NAMES = NO
+
+# If the SORT_BY_SCOPE_NAME tag is set to YES, the class list will be sorted by
+# fully-qualified names, including namespaces. If set to NO, the class list will
+# be sorted only by class name, not including the namespace part.
+# Note: This option is not very useful if HIDE_SCOPE_NAMES is set to YES.
+# Note: This option applies only to the class list, not to the alphabetical
+# list.
+# The default value is: NO.
+
+SORT_BY_SCOPE_NAME = YES
+
+# If the STRICT_PROTO_MATCHING option is enabled and doxygen fails to do proper
+# type resolution of all parameters of a function it will reject a match between
+# the prototype and the implementation of a member function even if there is
+# only one candidate or it is obvious which candidate to choose by doing a
+# simple string match. By disabling STRICT_PROTO_MATCHING doxygen will still
+# accept a match between prototype and implementation in such cases.
+# The default value is: NO.
+
+STRICT_PROTO_MATCHING = NO
+
+# The GENERATE_TODOLIST tag can be used to enable (YES) or disable (NO) the todo
+# list. This list is created by putting \todo commands in the documentation.
+# The default value is: YES.
+
+GENERATE_TODOLIST = NO
+
+# The GENERATE_TESTLIST tag can be used to enable (YES) or disable (NO) the test
+# list. This list is created by putting \test commands in the documentation.
+# The default value is: YES.
+
+GENERATE_TESTLIST = NO
+
+# The GENERATE_BUGLIST tag can be used to enable (YES) or disable (NO) the bug
+# list. This list is created by putting \bug commands in the documentation.
+# The default value is: YES.
+
+GENERATE_BUGLIST = NO
+
+# The GENERATE_DEPRECATEDLIST tag can be used to enable (YES) or disable (NO)
+# the deprecated list. This list is created by putting \deprecated commands in
+# the documentation.
+# The default value is: YES.
+
+GENERATE_DEPRECATEDLIST= YES
+
+# The ENABLED_SECTIONS tag can be used to enable conditional documentation
+# sections, marked by \if <section_label> ... \endif and \cond <section_label>
+# ... \endcond blocks.
+
+ENABLED_SECTIONS =
+
+# The MAX_INITIALIZER_LINES tag determines the maximum number of lines that the
+# initial value of a variable or macro / define can have for it to appear in the
+# documentation. If the initializer consists of more lines than specified here
+# it will be hidden. Use a value of 0 to hide initializers completely. The
+# appearance of the value of individual variables and macros / defines can be
+# controlled using \showinitializer or \hideinitializer command in the
+# documentation regardless of this setting.
+# Minimum value: 0, maximum value: 10000, default value: 30.
+
+MAX_INITIALIZER_LINES = 30
+
+# Set the SHOW_USED_FILES tag to NO to disable the list of files generated at
+# the bottom of the documentation of classes and structs. If set to YES, the
+# list will mention the files that were used to generate the documentation.
+# The default value is: YES.
+
+SHOW_USED_FILES = YES
+
+# Set the SHOW_FILES tag to NO to disable the generation of the Files page. This
+# will remove the Files entry from the Quick Index and from the Folder Tree View
+# (if specified).
+# The default value is: YES.
+
+SHOW_FILES = NO
+
+# Set the SHOW_NAMESPACES tag to NO to disable the generation of the Namespaces
+# page. This will remove the Namespaces entry from the Quick Index and from the
+# Folder Tree View (if specified).
+# The default value is: YES.
+
+SHOW_NAMESPACES = NO
+
+# The FILE_VERSION_FILTER tag can be used to specify a program or script that
+# doxygen should invoke to get the current version for each file (typically from
+# the version control system). Doxygen will invoke the program by executing (via
+# popen()) the command command input-file, where command is the value of the
+# FILE_VERSION_FILTER tag, and input-file is the name of an input file provided
+# by doxygen. Whatever the program writes to standard output is used as the file
+# version. For an example see the documentation.
+
+FILE_VERSION_FILTER =
+
+# The LAYOUT_FILE tag can be used to specify a layout file which will be parsed
+# by doxygen. The layout file controls the global structure of the generated
+# output files in an output format independent way. To create the layout file
+# that represents doxygen's defaults, run doxygen with the -l option. You can
+# optionally specify a file name after the option, if omitted DoxygenLayout.xml
+# will be used as the name of the layout file.
+#
+# Note that if you run doxygen from a directory containing a file called
+# DoxygenLayout.xml, doxygen will parse it automatically even if the LAYOUT_FILE
+# tag is left empty.
+
+LAYOUT_FILE = @SUIL_SRCDIR@/doc/layout.xml
+
+# The CITE_BIB_FILES tag can be used to specify one or more bib files containing
+# the reference definitions. This must be a list of .bib files. The .bib
+# extension is automatically appended if omitted. This requires the bibtex tool
+# to be installed. See also http://en.wikipedia.org/wiki/BibTeX for more info.
+# For LaTeX the style of the bibliography can be controlled using
+# LATEX_BIB_STYLE. To use this feature you need bibtex and perl available in the
+# search path. See also \cite for info how to create references.
+
+CITE_BIB_FILES =
+
+#---------------------------------------------------------------------------
+# Configuration options related to warning and progress messages
+#---------------------------------------------------------------------------
+
+# The QUIET tag can be used to turn on/off the messages that are generated to
+# standard output by doxygen. If QUIET is set to YES this implies that the
+# messages are off.
+# The default value is: NO.
+
+QUIET = YES
+
+# The WARNINGS tag can be used to turn on/off the warning messages that are
+# generated to standard error (stderr) by doxygen. If WARNINGS is set to YES
+# this implies that the warnings are on.
+#
+# Tip: Turn warnings on while writing the documentation.
+# The default value is: YES.
+
+WARNINGS = YES
+
+# If the WARN_IF_UNDOCUMENTED tag is set to YES then doxygen will generate
+# warnings for undocumented members. If EXTRACT_ALL is set to YES then this flag
+# will automatically be disabled.
+# The default value is: YES.
+
+WARN_IF_UNDOCUMENTED = YES
+
+# If the WARN_IF_DOC_ERROR tag is set to YES, doxygen will generate warnings for
+# potential errors in the documentation, such as not documenting some parameters
+# in a documented function, or documenting parameters that don't exist or using
+# markup commands wrongly.
+# The default value is: YES.
+
+WARN_IF_DOC_ERROR = YES
+
+# This WARN_NO_PARAMDOC option can be enabled to get warnings for functions that
+# are documented, but have no documentation for their parameters or return
+# value. If set to NO, doxygen will only warn about wrong or incomplete
+# parameter documentation, but not about the absence of documentation.
+# The default value is: NO.
+
+WARN_NO_PARAMDOC = YES
+
+# If the WARN_AS_ERROR tag is set to YES then doxygen will immediately stop when
+# a warning is encountered.
+# The default value is: NO.
+
+WARN_AS_ERROR = NO
+
+# The WARN_FORMAT tag determines the format of the warning messages that doxygen
+# can produce. The string should contain the $file, $line, and $text tags, which
+# will be replaced by the file and line number from which the warning originated
+# and the warning text. Optionally the format may contain $version, which will
+# be replaced by the version of the file (if it could be obtained via
+# FILE_VERSION_FILTER)
+# The default value is: $file:$line: $text.
+
+WARN_FORMAT = "$file:$line: $text"
+
+# The WARN_LOGFILE tag can be used to specify a file to which warning and error
+# messages should be written. If left blank the output is written to standard
+# error (stderr).
+
+WARN_LOGFILE =
+
+#---------------------------------------------------------------------------
+# Configuration options related to the input files
+#---------------------------------------------------------------------------
+
+# The INPUT tag is used to specify the files and/or directories that contain
+# documented source files. You may enter file names like myfile.cpp or
+# directories like /usr/src/myproject. Separate the files or directories with
+# spaces. See also FILE_PATTERNS and EXTENSION_MAPPING
+# Note: If this tag is empty the current directory is searched.
+
+INPUT = @SUIL_SRCDIR@/suil/suil.h
+
+# This tag can be used to specify the character encoding of the source files
+# that doxygen parses. Internally doxygen uses the UTF-8 encoding. Doxygen uses
+# libiconv (or the iconv built into libc) for the transcoding. See the libiconv
+# documentation (see: http://www.gnu.org/software/libiconv) for the list of
+# possible encodings.
+# The default value is: UTF-8.
+
+INPUT_ENCODING = UTF-8
+
+# If the value of the INPUT tag contains directories, you can use the
+# FILE_PATTERNS tag to specify one or more wildcard patterns (like *.cpp and
+# *.h) to filter out the source-files in the directories.
+#
+# Note that for custom extensions or not directly supported extensions you also
+# need to set EXTENSION_MAPPING for the extension otherwise the files are not
+# read by doxygen.
+#
+# If left blank the following patterns are tested:*.c, *.cc, *.cxx, *.cpp,
+# *.c++, *.java, *.ii, *.ixx, *.ipp, *.i++, *.inl, *.idl, *.ddl, *.odl, *.h,
+# *.hh, *.hxx, *.hpp, *.h++, *.cs, *.d, *.php, *.php4, *.php5, *.phtml, *.inc,
+# *.m, *.markdown, *.md, *.mm, *.dox, *.py, *.pyw, *.f90, *.f95, *.f03, *.f08,
+# *.f, *.for, *.tcl, *.vhd, *.vhdl, *.ucf and *.qsf.
+
+FILE_PATTERNS =
+
+# The RECURSIVE tag can be used to specify whether or not subdirectories should
+# be searched for input files as well.
+# The default value is: NO.
+
+RECURSIVE = NO
+
+# The EXCLUDE tag can be used to specify files and/or directories that should be
+# excluded from the INPUT source files. This way you can easily exclude a
+# subdirectory from a directory tree whose root is specified with the INPUT tag.
+#
+# Note that relative paths are relative to the directory from which doxygen is
+# run.
+
+EXCLUDE =
+
+# The EXCLUDE_SYMLINKS tag can be used to select whether or not files or
+# directories that are symbolic links (a Unix file system feature) are excluded
+# from the input.
+# The default value is: NO.
+
+EXCLUDE_SYMLINKS = NO
+
+# If the value of the INPUT tag contains directories, you can use the
+# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude
+# certain files from those directories.
+#
+# Note that the wildcards are matched against the file with absolute path, so to
+# exclude all test directories for example use the pattern */test/*
+
+EXCLUDE_PATTERNS =
+
+# The EXCLUDE_SYMBOLS tag can be used to specify one or more symbol names
+# (namespaces, classes, functions, etc.) that should be excluded from the
+# output. The symbol name can be a fully qualified name, a word, or if the
+# wildcard * is used, a substring. Examples: ANamespace, AClass,
+# AClass::ANamespace, ANamespace::*Test
+#
+# Note that the wildcards are matched against the file with absolute path, so to
+# exclude all test directories use the pattern */test/*
+
+EXCLUDE_SYMBOLS =
+
+# The EXAMPLE_PATH tag can be used to specify one or more files or directories
+# that contain example code fragments that are included (see the \include
+# command).
+
+EXAMPLE_PATH =
+
+# If the value of the EXAMPLE_PATH tag contains directories, you can use the
+# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp and
+# *.h) to filter out the source-files in the directories. If left blank all
+# files are included.
+
+EXAMPLE_PATTERNS = *.c
+
+# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be
+# searched for input files to be used with the \include or \dontinclude commands
+# irrespective of the value of the RECURSIVE tag.
+# The default value is: NO.
+
+EXAMPLE_RECURSIVE = NO
+
+# The IMAGE_PATH tag can be used to specify one or more files or directories
+# that contain images that are to be included in the documentation (see the
+# \image command).
+
+IMAGE_PATH =
+
+# The INPUT_FILTER tag can be used to specify a program that doxygen should
+# invoke to filter for each input file. Doxygen will invoke the filter program
+# by executing (via popen()) the command:
+#
+# <filter> <input-file>
+#
+# where <filter> is the value of the INPUT_FILTER tag, and <input-file> is the
+# name of an input file. Doxygen will then use the output that the filter
+# program writes to standard output. If FILTER_PATTERNS is specified, this tag
+# will be ignored.
+#
+# Note that the filter must not add or remove lines; it is applied before the
+# code is scanned, but not when the output code is generated. If lines are added
+# or removed, the anchors will not be placed correctly.
+#
+# Note that for custom extensions or not directly supported extensions you also
+# need to set EXTENSION_MAPPING for the extension otherwise the files are not
+# properly processed by doxygen.
+
+INPUT_FILTER =
+
+# The FILTER_PATTERNS tag can be used to specify filters on a per file pattern
+# basis. Doxygen will compare the file name with each pattern and apply the
+# filter if there is a match. The filters are a list of the form: pattern=filter
+# (like *.cpp=my_cpp_filter). See INPUT_FILTER for further information on how
+# filters are used. If the FILTER_PATTERNS tag is empty or if none of the
+# patterns match the file name, INPUT_FILTER is applied.
+#
+# Note that for custom extensions or not directly supported extensions you also
+# need to set EXTENSION_MAPPING for the extension otherwise the files are not
+# properly processed by doxygen.
+
+FILTER_PATTERNS =
+
+# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using
+# INPUT_FILTER) will also be used to filter the input files that are used for
+# producing the source files to browse (i.e. when SOURCE_BROWSER is set to YES).
+# The default value is: NO.
+
+FILTER_SOURCE_FILES = NO
+
+# The FILTER_SOURCE_PATTERNS tag can be used to specify source filters per file
+# pattern. A pattern will override the setting for FILTER_PATTERN (if any) and
+# it is also possible to disable source filtering for a specific pattern using
+# *.ext= (so without naming a filter).
+# This tag requires that the tag FILTER_SOURCE_FILES is set to YES.
+
+FILTER_SOURCE_PATTERNS =
+
+# If the USE_MDFILE_AS_MAINPAGE tag refers to the name of a markdown file that
+# is part of the input, its contents will be placed on the main page
+# (index.html). This can be useful if you have a project on for instance GitHub
+# and want to reuse the introduction page also for the doxygen output.
+
+USE_MDFILE_AS_MAINPAGE =
+
+#---------------------------------------------------------------------------
+# Configuration options related to source browsing
+#---------------------------------------------------------------------------
+
+# If the SOURCE_BROWSER tag is set to YES then a list of source files will be
+# generated. Documented entities will be cross-referenced with these sources.
+#
+# Note: To get rid of all source code in the generated output, make sure that
+# also VERBATIM_HEADERS is set to NO.
+# The default value is: NO.
+
+SOURCE_BROWSER = NO
+
+# Setting the INLINE_SOURCES tag to YES will include the body of functions,
+# classes and enums directly into the documentation.
+# The default value is: NO.
+
+INLINE_SOURCES = NO
+
+# Setting the STRIP_CODE_COMMENTS tag to YES will instruct doxygen to hide any
+# special comment blocks from generated source code fragments. Normal C, C++ and
+# Fortran comments will always remain visible.
+# The default value is: YES.
+
+STRIP_CODE_COMMENTS = YES
+
+# If the REFERENCED_BY_RELATION tag is set to YES then for each documented
+# function all documented functions referencing it will be listed.
+# The default value is: NO.
+
+REFERENCED_BY_RELATION = YES
+
+# If the REFERENCES_RELATION tag is set to YES then for each documented function
+# all documented entities called/used by that function will be listed.
+# The default value is: NO.
+
+REFERENCES_RELATION = YES
+
+# If the REFERENCES_LINK_SOURCE tag is set to YES and SOURCE_BROWSER tag is set
+# to YES then the hyperlinks from functions in REFERENCES_RELATION and
+# REFERENCED_BY_RELATION lists will link to the source code. Otherwise they will
+# link to the documentation.
+# The default value is: YES.
+
+REFERENCES_LINK_SOURCE = YES
+
+# If SOURCE_TOOLTIPS is enabled (the default) then hovering a hyperlink in the
+# source code will show a tooltip with additional information such as prototype,
+# brief description and links to the definition and documentation. Since this
+# will make the HTML file larger and loading of large files a bit slower, you
+# can opt to disable this feature.
+# The default value is: YES.
+# This tag requires that the tag SOURCE_BROWSER is set to YES.
+
+SOURCE_TOOLTIPS = YES
+
+# If the USE_HTAGS tag is set to YES then the references to source code will
+# point to the HTML generated by the htags(1) tool instead of doxygen built-in
+# source browser. The htags tool is part of GNU's global source tagging system
+# (see http://www.gnu.org/software/global/global.html). You will need version
+# 4.8.6 or higher.
+#
+# To use it do the following:
+# - Install the latest version of global
+# - Enable SOURCE_BROWSER and USE_HTAGS in the config file
+# - Make sure the INPUT points to the root of the source tree
+# - Run doxygen as normal
+#
+# Doxygen will invoke htags (and that will in turn invoke gtags), so these
+# tools must be available from the command line (i.e. in the search path).
+#
+# The result: instead of the source browser generated by doxygen, the links to
+# source code will now point to the output of htags.
+# The default value is: NO.
+# This tag requires that the tag SOURCE_BROWSER is set to YES.
+
+USE_HTAGS = NO
+
+# If the VERBATIM_HEADERS tag is set the YES then doxygen will generate a
+# verbatim copy of the header file for each class for which an include is
+# specified. Set to NO to disable this.
+# See also: Section \class.
+# The default value is: YES.
+
+VERBATIM_HEADERS = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the alphabetical class index
+#---------------------------------------------------------------------------
+
+# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index of all
+# compounds will be generated. Enable this if the project contains a lot of
+# classes, structs, unions or interfaces.
+# The default value is: YES.
+
+ALPHABETICAL_INDEX = NO
+
+# The COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns in
+# which the alphabetical index list will be split.
+# Minimum value: 1, maximum value: 20, default value: 5.
+# This tag requires that the tag ALPHABETICAL_INDEX is set to YES.
+
+COLS_IN_ALPHA_INDEX = 5
+
+# In case all classes in a project start with a common prefix, all classes will
+# be put under the same header in the alphabetical index. The IGNORE_PREFIX tag
+# can be used to specify a prefix (or a list of prefixes) that should be ignored
+# while generating the index headers.
+# This tag requires that the tag ALPHABETICAL_INDEX is set to YES.
+
+IGNORE_PREFIX =
+
+#---------------------------------------------------------------------------
+# Configuration options related to the HTML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_HTML tag is set to YES, doxygen will generate HTML output
+# The default value is: YES.
+
+GENERATE_HTML = YES
+
+# The HTML_OUTPUT tag is used to specify where the HTML docs will be put. If a
+# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of
+# it.
+# The default directory is: html.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_OUTPUT = html
+
+# The HTML_FILE_EXTENSION tag can be used to specify the file extension for each
+# generated HTML page (for example: .htm, .php, .asp).
+# The default value is: .html.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_FILE_EXTENSION = .html
+
+# The HTML_HEADER tag can be used to specify a user-defined HTML header file for
+# each generated HTML page. If the tag is left blank doxygen will generate a
+# standard header.
+#
+# To get valid HTML the header file that includes any scripts and style sheets
+# that doxygen needs, which is dependent on the configuration options used (e.g.
+# the setting GENERATE_TREEVIEW). It is highly recommended to start with a
+# default header using
+# doxygen -w html new_header.html new_footer.html new_stylesheet.css
+# YourConfigFile
+# and then modify the file new_header.html. See also section "Doxygen usage"
+# for information on how to generate the default header that doxygen normally
+# uses.
+# Note: The header is subject to change so you typically have to regenerate the
+# default header when upgrading to a newer version of doxygen. For a description
+# of the possible markers and block names see the documentation.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_HEADER =
+
+# The HTML_FOOTER tag can be used to specify a user-defined HTML footer for each
+# generated HTML page. If the tag is left blank doxygen will generate a standard
+# footer. See HTML_HEADER for more information on how to generate a default
+# footer and what special commands can be used inside the footer. See also
+# section "Doxygen usage" for information on how to generate the default footer
+# that doxygen normally uses.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_FOOTER =
+
+# The HTML_STYLESHEET tag can be used to specify a user-defined cascading style
+# sheet that is used by each HTML page. It can be used to fine-tune the look of
+# the HTML output. If left blank doxygen will generate a default style sheet.
+# See also section "Doxygen usage" for information on how to generate the style
+# sheet that doxygen normally uses.
+# Note: It is recommended to use HTML_EXTRA_STYLESHEET instead of this tag, as
+# it is more robust and this tag (HTML_STYLESHEET) will in the future become
+# obsolete.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_STYLESHEET = @SUIL_SRCDIR@/doc/style.css
+
+# The HTML_EXTRA_STYLESHEET tag can be used to specify additional user-defined
+# cascading style sheets that are included after the standard style sheets
+# created by doxygen. Using this option one can overrule certain style aspects.
+# This is preferred over using HTML_STYLESHEET since it does not replace the
+# standard style sheet and is therefore more robust against future updates.
+# Doxygen will copy the style sheet files to the output directory.
+# Note: The order of the extra style sheet files is of importance (e.g. the last
+# style sheet in the list overrules the setting of the previous ones in the
+# list). For an example see the documentation.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_EXTRA_STYLESHEET =
+
+# The HTML_EXTRA_FILES tag can be used to specify one or more extra images or
+# other source files which should be copied to the HTML output directory. Note
+# that these files will be copied to the base HTML output directory. Use the
+# $relpath^ marker in the HTML_HEADER and/or HTML_FOOTER files to load these
+# files. In the HTML_STYLESHEET file, use the file name only. Also note that the
+# files will be copied as-is; there are no commands or markers available.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_EXTRA_FILES =
+
+# The HTML_COLORSTYLE_HUE tag controls the color of the HTML output. Doxygen
+# will adjust the colors in the style sheet and background images according to
+# this color. Hue is specified as an angle on a colorwheel, see
+# http://en.wikipedia.org/wiki/Hue for more information. For instance the value
+# 0 represents red, 60 is yellow, 120 is green, 180 is cyan, 240 is blue, 300
+# purple, and 360 is red again.
+# Minimum value: 0, maximum value: 359, default value: 220.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_COLORSTYLE_HUE = 160
+
+# The HTML_COLORSTYLE_SAT tag controls the purity (or saturation) of the colors
+# in the HTML output. For a value of 0 the output will use grayscales only. A
+# value of 255 will produce the most vivid colors.
+# Minimum value: 0, maximum value: 255, default value: 100.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_COLORSTYLE_SAT = 100
+
+# The HTML_COLORSTYLE_GAMMA tag controls the gamma correction applied to the
+# luminance component of the colors in the HTML output. Values below 100
+# gradually make the output lighter, whereas values above 100 make the output
+# darker. The value divided by 100 is the actual gamma applied, so 80 represents
+# a gamma of 0.8, The value 220 represents a gamma of 2.2, and 100 does not
+# change the gamma.
+# Minimum value: 40, maximum value: 240, default value: 80.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_COLORSTYLE_GAMMA = 80
+
+# If the HTML_TIMESTAMP tag is set to YES then the footer of each generated HTML
+# page will contain the date and time when the page was generated. Setting this
+# to YES can help to show when doxygen was last run and thus if the
+# documentation is up to date.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_TIMESTAMP = NO
+
+# If the HTML_DYNAMIC_SECTIONS tag is set to YES then the generated HTML
+# documentation will contain sections that can be hidden and shown after the
+# page has loaded.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_DYNAMIC_SECTIONS = NO
+
+# With HTML_INDEX_NUM_ENTRIES one can control the preferred number of entries
+# shown in the various tree structured indices initially; the user can expand
+# and collapse entries dynamically later on. Doxygen will expand the tree to
+# such a level that at most the specified number of entries are visible (unless
+# a fully collapsed tree already exceeds this amount). So setting the number of
+# entries 1 will produce a full collapsed tree by default. 0 is a special value
+# representing an infinite number of entries and will result in a full expanded
+# tree by default.
+# Minimum value: 0, maximum value: 9999, default value: 100.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+HTML_INDEX_NUM_ENTRIES = 100
+
+# If the GENERATE_DOCSET tag is set to YES, additional index files will be
+# generated that can be used as input for Apple's Xcode 3 integrated development
+# environment (see: http://developer.apple.com/tools/xcode/), introduced with
+# OSX 10.5 (Leopard). To create a documentation set, doxygen will generate a
+# Makefile in the HTML output directory. Running make will produce the docset in
+# that directory and running make install will install the docset in
+# ~/Library/Developer/Shared/Documentation/DocSets so that Xcode will find it at
+# startup. See http://developer.apple.com/tools/creatingdocsetswithdoxygen.html
+# for more information.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+GENERATE_DOCSET = NO
+
+# This tag determines the name of the docset feed. A documentation feed provides
+# an umbrella under which multiple documentation sets from a single provider
+# (such as a company or product suite) can be grouped.
+# The default value is: Doxygen generated docs.
+# This tag requires that the tag GENERATE_DOCSET is set to YES.
+
+DOCSET_FEEDNAME = "Doxygen generated docs"
+
+# This tag specifies a string that should uniquely identify the documentation
+# set bundle. This should be a reverse domain-name style string, e.g.
+# com.mycompany.MyDocSet. Doxygen will append .docset to the name.
+# The default value is: org.doxygen.Project.
+# This tag requires that the tag GENERATE_DOCSET is set to YES.
+
+DOCSET_BUNDLE_ID = org.doxygen.Project
+
+# The DOCSET_PUBLISHER_ID tag specifies a string that should uniquely identify
+# the documentation publisher. This should be a reverse domain-name style
+# string, e.g. com.mycompany.MyDocSet.documentation.
+# The default value is: org.doxygen.Publisher.
+# This tag requires that the tag GENERATE_DOCSET is set to YES.
+
+DOCSET_PUBLISHER_ID = org.doxygen.Publisher
+
+# The DOCSET_PUBLISHER_NAME tag identifies the documentation publisher.
+# The default value is: Publisher.
+# This tag requires that the tag GENERATE_DOCSET is set to YES.
+
+DOCSET_PUBLISHER_NAME = Publisher
+
+# If the GENERATE_HTMLHELP tag is set to YES then doxygen generates three
+# additional HTML index files: index.hhp, index.hhc, and index.hhk. The
+# index.hhp is a project file that can be read by Microsoft's HTML Help Workshop
+# (see: http://www.microsoft.com/en-us/download/details.aspx?id=21138) on
+# Windows.
+#
+# The HTML Help Workshop contains a compiler that can convert all HTML output
+# generated by doxygen into a single compiled HTML file (.chm). Compiled HTML
+# files are now used as the Windows 98 help format, and will replace the old
+# Windows help format (.hlp) on all Windows platforms in the future. Compressed
+# HTML files also contain an index, a table of contents, and you can search for
+# words in the documentation. The HTML workshop also contains a viewer for
+# compressed HTML files.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+GENERATE_HTMLHELP = NO
+
+# The CHM_FILE tag can be used to specify the file name of the resulting .chm
+# file. You can add a path in front of the file if the result should not be
+# written to the html output directory.
+# This tag requires that the tag GENERATE_HTMLHELP is set to YES.
+
+CHM_FILE =
+
+# The HHC_LOCATION tag can be used to specify the location (absolute path
+# including file name) of the HTML help compiler (hhc.exe). If non-empty,
+# doxygen will try to run the HTML help compiler on the generated index.hhp.
+# The file has to be specified with full path.
+# This tag requires that the tag GENERATE_HTMLHELP is set to YES.
+
+HHC_LOCATION =
+
+# The GENERATE_CHI flag controls if a separate .chi index file is generated
+# (YES) or that it should be included in the master .chm file (NO).
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTMLHELP is set to YES.
+
+GENERATE_CHI = NO
+
+# The CHM_INDEX_ENCODING is used to encode HtmlHelp index (hhk), content (hhc)
+# and project file content.
+# This tag requires that the tag GENERATE_HTMLHELP is set to YES.
+
+CHM_INDEX_ENCODING =
+
+# The BINARY_TOC flag controls whether a binary table of contents is generated
+# (YES) or a normal table of contents (NO) in the .chm file. Furthermore it
+# enables the Previous and Next buttons.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTMLHELP is set to YES.
+
+BINARY_TOC = NO
+
+# The TOC_EXPAND flag can be set to YES to add extra items for group members to
+# the table of contents of the HTML help documentation and to the tree view.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTMLHELP is set to YES.
+
+TOC_EXPAND = NO
+
+# If the GENERATE_QHP tag is set to YES and both QHP_NAMESPACE and
+# QHP_VIRTUAL_FOLDER are set, an additional index file will be generated that
+# can be used as input for Qt's qhelpgenerator to generate a Qt Compressed Help
+# (.qch) of the generated HTML documentation.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+GENERATE_QHP = NO
+
+# If the QHG_LOCATION tag is specified, the QCH_FILE tag can be used to specify
+# the file name of the resulting .qch file. The path specified is relative to
+# the HTML output folder.
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QCH_FILE =
+
+# The QHP_NAMESPACE tag specifies the namespace to use when generating Qt Help
+# Project output. For more information please see Qt Help Project / Namespace
+# (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#namespace).
+# The default value is: org.doxygen.Project.
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QHP_NAMESPACE =
+
+# The QHP_VIRTUAL_FOLDER tag specifies the namespace to use when generating Qt
+# Help Project output. For more information please see Qt Help Project / Virtual
+# Folders (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#virtual-
+# folders).
+# The default value is: doc.
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QHP_VIRTUAL_FOLDER = doc
+
+# If the QHP_CUST_FILTER_NAME tag is set, it specifies the name of a custom
+# filter to add. For more information please see Qt Help Project / Custom
+# Filters (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#custom-
+# filters).
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QHP_CUST_FILTER_NAME =
+
+# The QHP_CUST_FILTER_ATTRS tag specifies the list of the attributes of the
+# custom filter to add. For more information please see Qt Help Project / Custom
+# Filters (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#custom-
+# filters).
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QHP_CUST_FILTER_ATTRS =
+
+# The QHP_SECT_FILTER_ATTRS tag specifies the list of the attributes this
+# project's filter section matches. Qt Help Project / Filter Attributes (see:
+# http://qt-project.org/doc/qt-4.8/qthelpproject.html#filter-attributes).
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QHP_SECT_FILTER_ATTRS =
+
+# The QHG_LOCATION tag can be used to specify the location of Qt's
+# qhelpgenerator. If non-empty doxygen will try to run qhelpgenerator on the
+# generated .qhp file.
+# This tag requires that the tag GENERATE_QHP is set to YES.
+
+QHG_LOCATION =
+
+# If the GENERATE_ECLIPSEHELP tag is set to YES, additional index files will be
+# generated, together with the HTML files, they form an Eclipse help plugin. To
+# install this plugin and make it available under the help contents menu in
+# Eclipse, the contents of the directory containing the HTML and XML files needs
+# to be copied into the plugins directory of eclipse. The name of the directory
+# within the plugins directory should be the same as the ECLIPSE_DOC_ID value.
+# After copying Eclipse needs to be restarted before the help appears.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+GENERATE_ECLIPSEHELP = NO
+
+# A unique identifier for the Eclipse help plugin. When installing the plugin
+# the directory name containing the HTML and XML files should also have this
+# name. Each documentation set should have its own identifier.
+# The default value is: org.doxygen.Project.
+# This tag requires that the tag GENERATE_ECLIPSEHELP is set to YES.
+
+ECLIPSE_DOC_ID = org.doxygen.Project
+
+# If you want full control over the layout of the generated HTML pages it might
+# be necessary to disable the index and replace it with your own. The
+# DISABLE_INDEX tag can be used to turn on/off the condensed index (tabs) at top
+# of each HTML page. A value of NO enables the index and the value YES disables
+# it. Since the tabs in the index contain the same information as the navigation
+# tree, you can set this option to YES if you also set GENERATE_TREEVIEW to YES.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+DISABLE_INDEX = YES
+
+# The GENERATE_TREEVIEW tag is used to specify whether a tree-like index
+# structure should be generated to display hierarchical information. If the tag
+# value is set to YES, a side panel will be generated containing a tree-like
+# index structure (just like the one that is generated for HTML Help). For this
+# to work a browser that supports JavaScript, DHTML, CSS and frames is required
+# (i.e. any modern browser). Windows users are probably better off using the
+# HTML help feature. Via custom style sheets (see HTML_EXTRA_STYLESHEET) one can
+# further fine-tune the look of the index. As an example, the default style
+# sheet generated by doxygen has an example that shows how to put an image at
+# the root of the tree instead of the PROJECT_NAME. Since the tree basically has
+# the same information as the tab index, you could consider setting
+# DISABLE_INDEX to YES when enabling this option.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+GENERATE_TREEVIEW = NO
+
+# The ENUM_VALUES_PER_LINE tag can be used to set the number of enum values that
+# doxygen will group on one line in the generated HTML documentation.
+#
+# Note that a value of 0 will completely suppress the enum values from appearing
+# in the overview section.
+# Minimum value: 0, maximum value: 20, default value: 4.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+ENUM_VALUES_PER_LINE = 1
+
+# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be used
+# to set the initial width (in pixels) of the frame in which the tree is shown.
+# Minimum value: 0, maximum value: 1500, default value: 250.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+TREEVIEW_WIDTH = 250
+
+# If the EXT_LINKS_IN_WINDOW option is set to YES, doxygen will open links to
+# external symbols imported via tag files in a separate window.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+EXT_LINKS_IN_WINDOW = NO
+
+# Use this tag to change the font size of LaTeX formulas included as images in
+# the HTML documentation. When you change the font size after a successful
+# doxygen run you need to manually remove any form_*.png images from the HTML
+# output directory to force them to be regenerated.
+# Minimum value: 8, maximum value: 50, default value: 10.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+FORMULA_FONTSIZE = 10
+
+# Use the FORMULA_TRANPARENT tag to determine whether or not the images
+# generated for formulas are transparent PNGs. Transparent PNGs are not
+# supported properly for IE 6.0, but are supported on all modern browsers.
+#
+# Note that when changing this option you need to delete any form_*.png files in
+# the HTML output directory before the changes have effect.
+# The default value is: YES.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+FORMULA_TRANSPARENT = YES
+
+# Enable the USE_MATHJAX option to render LaTeX formulas using MathJax (see
+# http://www.mathjax.org) which uses client side Javascript for the rendering
+# instead of using pre-rendered bitmaps. Use this if you do not have LaTeX
+# installed or if you want to formulas look prettier in the HTML output. When
+# enabled you may also need to install MathJax separately and configure the path
+# to it using the MATHJAX_RELPATH option.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+USE_MATHJAX = NO
+
+# When MathJax is enabled you can set the default output format to be used for
+# the MathJax output. See the MathJax site (see:
+# http://docs.mathjax.org/en/latest/output.html) for more details.
+# Possible values are: HTML-CSS (which is slower, but has the best
+# compatibility), NativeMML (i.e. MathML) and SVG.
+# The default value is: HTML-CSS.
+# This tag requires that the tag USE_MATHJAX is set to YES.
+
+MATHJAX_FORMAT = HTML-CSS
+
+# When MathJax is enabled you need to specify the location relative to the HTML
+# output directory using the MATHJAX_RELPATH option. The destination directory
+# should contain the MathJax.js script. For instance, if the mathjax directory
+# is located at the same level as the HTML output directory, then
+# MATHJAX_RELPATH should be ../mathjax. The default value points to the MathJax
+# Content Delivery Network so you can quickly see the result without installing
+# MathJax. However, it is strongly recommended to install a local copy of
+# MathJax from http://www.mathjax.org before deployment.
+# The default value is: http://cdn.mathjax.org/mathjax/latest.
+# This tag requires that the tag USE_MATHJAX is set to YES.
+
+MATHJAX_RELPATH = http://cdn.mathjax.org/mathjax/latest
+
+# The MATHJAX_EXTENSIONS tag can be used to specify one or more MathJax
+# extension names that should be enabled during MathJax rendering. For example
+# MATHJAX_EXTENSIONS = TeX/AMSmath TeX/AMSsymbols
+# This tag requires that the tag USE_MATHJAX is set to YES.
+
+MATHJAX_EXTENSIONS =
+
+# The MATHJAX_CODEFILE tag can be used to specify a file with javascript pieces
+# of code that will be used on startup of the MathJax code. See the MathJax site
+# (see: http://docs.mathjax.org/en/latest/output.html) for more details. For an
+# example see the documentation.
+# This tag requires that the tag USE_MATHJAX is set to YES.
+
+MATHJAX_CODEFILE =
+
+# When the SEARCHENGINE tag is enabled doxygen will generate a search box for
+# the HTML output. The underlying search engine uses javascript and DHTML and
+# should work on any modern browser. Note that when using HTML help
+# (GENERATE_HTMLHELP), Qt help (GENERATE_QHP), or docsets (GENERATE_DOCSET)
+# there is already a search function so this one should typically be disabled.
+# For large projects the javascript based search engine can be slow, then
+# enabling SERVER_BASED_SEARCH may provide a better solution. It is possible to
+# search using the keyboard; to jump to the search box use <access key> + S
+# (what the <access key> is depends on the OS and browser, but it is typically
+# <CTRL>, <ALT>/<option>, or both). Inside the search box use the <cursor down
+# key> to jump into the search results window, the results can be navigated
+# using the <cursor keys>. Press <Enter> to select an item or <escape> to cancel
+# the search. The filter options can be selected when the cursor is inside the
+# search box by pressing <Shift>+<cursor down>. Also here use the <cursor keys>
+# to select a filter and <Enter> or <escape> to activate or cancel the filter
+# option.
+# The default value is: YES.
+# This tag requires that the tag GENERATE_HTML is set to YES.
+
+SEARCHENGINE = NO
+
+# When the SERVER_BASED_SEARCH tag is enabled the search engine will be
+# implemented using a web server instead of a web client using Javascript. There
+# are two flavors of web server based searching depending on the EXTERNAL_SEARCH
+# setting. When disabled, doxygen will generate a PHP script for searching and
+# an index file used by the script. When EXTERNAL_SEARCH is enabled the indexing
+# and searching needs to be provided by external tools. See the section
+# "External Indexing and Searching" for details.
+# The default value is: NO.
+# This tag requires that the tag SEARCHENGINE is set to YES.
+
+SERVER_BASED_SEARCH = NO
+
+# When EXTERNAL_SEARCH tag is enabled doxygen will no longer generate the PHP
+# script for searching. Instead the search results are written to an XML file
+# which needs to be processed by an external indexer. Doxygen will invoke an
+# external search engine pointed to by the SEARCHENGINE_URL option to obtain the
+# search results.
+#
+# Doxygen ships with an example indexer (doxyindexer) and search engine
+# (doxysearch.cgi) which are based on the open source search engine library
+# Xapian (see: http://xapian.org/).
+#
+# See the section "External Indexing and Searching" for details.
+# The default value is: NO.
+# This tag requires that the tag SEARCHENGINE is set to YES.
+
+EXTERNAL_SEARCH = NO
+
+# The SEARCHENGINE_URL should point to a search engine hosted by a web server
+# which will return the search results when EXTERNAL_SEARCH is enabled.
+#
+# Doxygen ships with an example indexer (doxyindexer) and search engine
+# (doxysearch.cgi) which are based on the open source search engine library
+# Xapian (see: http://xapian.org/). See the section "External Indexing and
+# Searching" for details.
+# This tag requires that the tag SEARCHENGINE is set to YES.
+
+SEARCHENGINE_URL =
+
+# When SERVER_BASED_SEARCH and EXTERNAL_SEARCH are both enabled the unindexed
+# search data is written to a file for indexing by an external tool. With the
+# SEARCHDATA_FILE tag the name of this file can be specified.
+# The default file is: searchdata.xml.
+# This tag requires that the tag SEARCHENGINE is set to YES.
+
+SEARCHDATA_FILE = searchdata.xml
+
+# When SERVER_BASED_SEARCH and EXTERNAL_SEARCH are both enabled the
+# EXTERNAL_SEARCH_ID tag can be used as an identifier for the project. This is
+# useful in combination with EXTRA_SEARCH_MAPPINGS to search through multiple
+# projects and redirect the results back to the right project.
+# This tag requires that the tag SEARCHENGINE is set to YES.
+
+EXTERNAL_SEARCH_ID =
+
+# The EXTRA_SEARCH_MAPPINGS tag can be used to enable searching through doxygen
+# projects other than the one defined by this configuration file, but that are
+# all added to the same external search index. Each project needs to have a
+# unique id set via EXTERNAL_SEARCH_ID. The search mapping then maps the id of
+# to a relative location where the documentation can be found. The format is:
+# EXTRA_SEARCH_MAPPINGS = tagname1=loc1 tagname2=loc2 ...
+# This tag requires that the tag SEARCHENGINE is set to YES.
+
+EXTRA_SEARCH_MAPPINGS =
+
+#---------------------------------------------------------------------------
+# Configuration options related to the LaTeX output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_LATEX tag is set to YES, doxygen will generate LaTeX output.
+# The default value is: YES.
+
+GENERATE_LATEX = NO
+
+# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put. If a
+# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of
+# it.
+# The default directory is: latex.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_OUTPUT = latex
+
+# The LATEX_CMD_NAME tag can be used to specify the LaTeX command name to be
+# invoked.
+#
+# Note that when enabling USE_PDFLATEX this option is only used for generating
+# bitmaps for formulas in the HTML output, but not in the Makefile that is
+# written to the output directory.
+# The default file is: latex.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_CMD_NAME = latex
+
+# The MAKEINDEX_CMD_NAME tag can be used to specify the command name to generate
+# index for LaTeX.
+# The default file is: makeindex.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+MAKEINDEX_CMD_NAME = makeindex
+
+# If the COMPACT_LATEX tag is set to YES, doxygen generates more compact LaTeX
+# documents. This may be useful for small projects and may help to save some
+# trees in general.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+COMPACT_LATEX = NO
+
+# The PAPER_TYPE tag can be used to set the paper type that is used by the
+# printer.
+# Possible values are: a4 (210 x 297 mm), letter (8.5 x 11 inches), legal (8.5 x
+# 14 inches) and executive (7.25 x 10.5 inches).
+# The default value is: a4.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+PAPER_TYPE = a4wide
+
+# The EXTRA_PACKAGES tag can be used to specify one or more LaTeX package names
+# that should be included in the LaTeX output. The package can be specified just
+# by its name or with the correct syntax as to be used with the LaTeX
+# \usepackage command. To get the times font for instance you can specify :
+# EXTRA_PACKAGES=times or EXTRA_PACKAGES={times}
+# To use the option intlimits with the amsmath package you can specify:
+# EXTRA_PACKAGES=[intlimits]{amsmath}
+# If left blank no extra packages will be included.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+EXTRA_PACKAGES =
+
+# The LATEX_HEADER tag can be used to specify a personal LaTeX header for the
+# generated LaTeX document. The header should contain everything until the first
+# chapter. If it is left blank doxygen will generate a standard header. See
+# section "Doxygen usage" for information on how to let doxygen write the
+# default header to a separate file.
+#
+# Note: Only use a user-defined header if you know what you are doing! The
+# following commands have a special meaning inside the header: $title,
+# $datetime, $date, $doxygenversion, $projectname, $projectnumber,
+# $projectbrief, $projectlogo. Doxygen will replace $title with the empty
+# string, for the replacement values of the other commands the user is referred
+# to HTML_HEADER.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_HEADER =
+
+# The LATEX_FOOTER tag can be used to specify a personal LaTeX footer for the
+# generated LaTeX document. The footer should contain everything after the last
+# chapter. If it is left blank doxygen will generate a standard footer. See
+# LATEX_HEADER for more information on how to generate a default footer and what
+# special commands can be used inside the footer.
+#
+# Note: Only use a user-defined footer if you know what you are doing!
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_FOOTER =
+
+# The LATEX_EXTRA_STYLESHEET tag can be used to specify additional user-defined
+# LaTeX style sheets that are included after the standard style sheets created
+# by doxygen. Using this option one can overrule certain style aspects. Doxygen
+# will copy the style sheet files to the output directory.
+# Note: The order of the extra style sheet files is of importance (e.g. the last
+# style sheet in the list overrules the setting of the previous ones in the
+# list).
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_EXTRA_STYLESHEET =
+
+# The LATEX_EXTRA_FILES tag can be used to specify one or more extra images or
+# other source files which should be copied to the LATEX_OUTPUT output
+# directory. Note that the files will be copied as-is; there are no commands or
+# markers available.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_EXTRA_FILES =
+
+# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated is
+# prepared for conversion to PDF (using ps2pdf or pdflatex). The PDF file will
+# contain links (just like the HTML output) instead of page references. This
+# makes the output suitable for online browsing using a PDF viewer.
+# The default value is: YES.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+PDF_HYPERLINKS = YES
+
+# If the USE_PDFLATEX tag is set to YES, doxygen will use pdflatex to generate
+# the PDF file directly from the LaTeX files. Set this option to YES, to get a
+# higher quality PDF documentation.
+# The default value is: YES.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+USE_PDFLATEX = YES
+
+# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \batchmode
+# command to the generated LaTeX files. This will instruct LaTeX to keep running
+# if errors occur, instead of asking the user for help. This option is also used
+# when generating formulas in HTML.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_BATCHMODE = NO
+
+# If the LATEX_HIDE_INDICES tag is set to YES then doxygen will not include the
+# index chapters (such as File Index, Compound Index, etc.) in the output.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_HIDE_INDICES = NO
+
+# If the LATEX_SOURCE_CODE tag is set to YES then doxygen will include source
+# code with syntax highlighting in the LaTeX output.
+#
+# Note that which sources are shown also depends on other settings such as
+# SOURCE_BROWSER.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_SOURCE_CODE = NO
+
+# The LATEX_BIB_STYLE tag can be used to specify the style to use for the
+# bibliography, e.g. plainnat, or ieeetr. See
+# http://en.wikipedia.org/wiki/BibTeX and \cite for more info.
+# The default value is: plain.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_BIB_STYLE = plain
+
+# If the LATEX_TIMESTAMP tag is set to YES then the footer of each generated
+# page will contain the date and time when the page was generated. Setting this
+# to NO can help when comparing the output of multiple runs.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_LATEX is set to YES.
+
+LATEX_TIMESTAMP = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the RTF output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_RTF tag is set to YES, doxygen will generate RTF output. The
+# RTF output is optimized for Word 97 and may not look too pretty with other RTF
+# readers/editors.
+# The default value is: NO.
+
+GENERATE_RTF = NO
+
+# The RTF_OUTPUT tag is used to specify where the RTF docs will be put. If a
+# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of
+# it.
+# The default directory is: rtf.
+# This tag requires that the tag GENERATE_RTF is set to YES.
+
+RTF_OUTPUT = rtf
+
+# If the COMPACT_RTF tag is set to YES, doxygen generates more compact RTF
+# documents. This may be useful for small projects and may help to save some
+# trees in general.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_RTF is set to YES.
+
+COMPACT_RTF = NO
+
+# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated will
+# contain hyperlink fields. The RTF file will contain links (just like the HTML
+# output) instead of page references. This makes the output suitable for online
+# browsing using Word or some other Word compatible readers that support those
+# fields.
+#
+# Note: WordPad (write) and others do not support links.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_RTF is set to YES.
+
+RTF_HYPERLINKS = NO
+
+# Load stylesheet definitions from file. Syntax is similar to doxygen's config
+# file, i.e. a series of assignments. You only have to provide replacements,
+# missing definitions are set to their default value.
+#
+# See also section "Doxygen usage" for information on how to generate the
+# default style sheet that doxygen normally uses.
+# This tag requires that the tag GENERATE_RTF is set to YES.
+
+RTF_STYLESHEET_FILE =
+
+# Set optional variables used in the generation of an RTF document. Syntax is
+# similar to doxygen's config file. A template extensions file can be generated
+# using doxygen -e rtf extensionFile.
+# This tag requires that the tag GENERATE_RTF is set to YES.
+
+RTF_EXTENSIONS_FILE =
+
+# If the RTF_SOURCE_CODE tag is set to YES then doxygen will include source code
+# with syntax highlighting in the RTF output.
+#
+# Note that which sources are shown also depends on other settings such as
+# SOURCE_BROWSER.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_RTF is set to YES.
+
+RTF_SOURCE_CODE = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the man page output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_MAN tag is set to YES, doxygen will generate man pages for
+# classes and files.
+# The default value is: NO.
+
+GENERATE_MAN = YES
+
+# The MAN_OUTPUT tag is used to specify where the man pages will be put. If a
+# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of
+# it. A directory man3 will be created inside the directory specified by
+# MAN_OUTPUT.
+# The default directory is: man.
+# This tag requires that the tag GENERATE_MAN is set to YES.
+
+MAN_OUTPUT = man
+
+# The MAN_EXTENSION tag determines the extension that is added to the generated
+# man pages. In case the manual section does not start with a number, the number
+# 3 is prepended. The dot (.) at the beginning of the MAN_EXTENSION tag is
+# optional.
+# The default value is: .3.
+# This tag requires that the tag GENERATE_MAN is set to YES.
+
+MAN_EXTENSION = .3
+
+# The MAN_SUBDIR tag determines the name of the directory created within
+# MAN_OUTPUT in which the man pages are placed. If defaults to man followed by
+# MAN_EXTENSION with the initial . removed.
+# This tag requires that the tag GENERATE_MAN is set to YES.
+
+MAN_SUBDIR =
+
+# If the MAN_LINKS tag is set to YES and doxygen generates man output, then it
+# will generate one additional man file for each entity documented in the real
+# man page(s). These additional files only source the real man page, but without
+# them the man command would be unable to find the correct page.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_MAN is set to YES.
+
+MAN_LINKS = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the XML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_XML tag is set to YES, doxygen will generate an XML file that
+# captures the structure of the code including all documentation.
+# The default value is: NO.
+
+GENERATE_XML = NO
+
+# The XML_OUTPUT tag is used to specify where the XML pages will be put. If a
+# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of
+# it.
+# The default directory is: xml.
+# This tag requires that the tag GENERATE_XML is set to YES.
+
+XML_OUTPUT = xml
+
+# If the XML_PROGRAMLISTING tag is set to YES, doxygen will dump the program
+# listings (including syntax highlighting and cross-referencing information) to
+# the XML output. Note that enabling this will significantly increase the size
+# of the XML output.
+# The default value is: YES.
+# This tag requires that the tag GENERATE_XML is set to YES.
+
+XML_PROGRAMLISTING = YES
+
+#---------------------------------------------------------------------------
+# Configuration options related to the DOCBOOK output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_DOCBOOK tag is set to YES, doxygen will generate Docbook files
+# that can be used to generate PDF.
+# The default value is: NO.
+
+GENERATE_DOCBOOK = NO
+
+# The DOCBOOK_OUTPUT tag is used to specify where the Docbook pages will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be put in
+# front of it.
+# The default directory is: docbook.
+# This tag requires that the tag GENERATE_DOCBOOK is set to YES.
+
+DOCBOOK_OUTPUT = docbook
+
+# If the DOCBOOK_PROGRAMLISTING tag is set to YES, doxygen will include the
+# program listings (including syntax highlighting and cross-referencing
+# information) to the DOCBOOK output. Note that enabling this will significantly
+# increase the size of the DOCBOOK output.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_DOCBOOK is set to YES.
+
+DOCBOOK_PROGRAMLISTING = NO
+
+#---------------------------------------------------------------------------
+# Configuration options for the AutoGen Definitions output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_AUTOGEN_DEF tag is set to YES, doxygen will generate an
+# AutoGen Definitions (see http://autogen.sf.net) file that captures the
+# structure of the code including all documentation. Note that this feature is
+# still experimental and incomplete at the moment.
+# The default value is: NO.
+
+GENERATE_AUTOGEN_DEF = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the Perl module output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_PERLMOD tag is set to YES, doxygen will generate a Perl module
+# file that captures the structure of the code including all documentation.
+#
+# Note that this feature is still experimental and incomplete at the moment.
+# The default value is: NO.
+
+GENERATE_PERLMOD = NO
+
+# If the PERLMOD_LATEX tag is set to YES, doxygen will generate the necessary
+# Makefile rules, Perl scripts and LaTeX code to be able to generate PDF and DVI
+# output from the Perl module output.
+# The default value is: NO.
+# This tag requires that the tag GENERATE_PERLMOD is set to YES.
+
+PERLMOD_LATEX = NO
+
+# If the PERLMOD_PRETTY tag is set to YES, the Perl module output will be nicely
+# formatted so it can be parsed by a human reader. This is useful if you want to
+# understand what is going on. On the other hand, if this tag is set to NO, the
+# size of the Perl module output will be much smaller and Perl will parse it
+# just the same.
+# The default value is: YES.
+# This tag requires that the tag GENERATE_PERLMOD is set to YES.
+
+PERLMOD_PRETTY = YES
+
+# The names of the make variables in the generated doxyrules.make file are
+# prefixed with the string contained in PERLMOD_MAKEVAR_PREFIX. This is useful
+# so different doxyrules.make files included by the same Makefile don't
+# overwrite each other's variables.
+# This tag requires that the tag GENERATE_PERLMOD is set to YES.
+
+PERLMOD_MAKEVAR_PREFIX =
+
+#---------------------------------------------------------------------------
+# Configuration options related to the preprocessor
+#---------------------------------------------------------------------------
+
+# If the ENABLE_PREPROCESSING tag is set to YES, doxygen will evaluate all
+# C-preprocessor directives found in the sources and include files.
+# The default value is: YES.
+
+ENABLE_PREPROCESSING = YES
+
+# If the MACRO_EXPANSION tag is set to YES, doxygen will expand all macro names
+# in the source code. If set to NO, only conditional compilation will be
+# performed. Macro expansion can be done in a controlled way by setting
+# EXPAND_ONLY_PREDEF to YES.
+# The default value is: NO.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+MACRO_EXPANSION = NO
+
+# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES then
+# the macro expansion is limited to the macros specified with the PREDEFINED and
+# EXPAND_AS_DEFINED tags.
+# The default value is: NO.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+EXPAND_ONLY_PREDEF = NO
+
+# If the SEARCH_INCLUDES tag is set to YES, the include files in the
+# INCLUDE_PATH will be searched if a #include is found.
+# The default value is: YES.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+SEARCH_INCLUDES = YES
+
+# The INCLUDE_PATH tag can be used to specify one or more directories that
+# contain include files that are not input files but should be processed by the
+# preprocessor.
+# This tag requires that the tag SEARCH_INCLUDES is set to YES.
+
+INCLUDE_PATH =
+
+# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard
+# patterns (like *.h and *.hpp) to filter out the header-files in the
+# directories. If left blank, the patterns specified with FILE_PATTERNS will be
+# used.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+INCLUDE_FILE_PATTERNS =
+
+# The PREDEFINED tag can be used to specify one or more macro names that are
+# defined before the preprocessor is started (similar to the -D option of e.g.
+# gcc). The argument of the tag is a list of macros of the form: name or
+# name=definition (no spaces). If the definition and the "=" are omitted, "=1"
+# is assumed. To prevent a macro definition from being undefined via #undef or
+# recursively expanded use the := operator instead of the = operator.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+PREDEFINED =
+
+# If the MACRO_EXPANSION and EXPAND_ONLY_PREDEF tags are set to YES then this
+# tag can be used to specify a list of macro names that should be expanded. The
+# macro definition that is found in the sources will be used. Use the PREDEFINED
+# tag if you want to use a different macro definition that overrules the
+# definition found in the source code.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+EXPAND_AS_DEFINED =
+
+# If the SKIP_FUNCTION_MACROS tag is set to YES then doxygen's preprocessor will
+# remove all references to function-like macros that are alone on a line, have
+# an all uppercase name, and do not end with a semicolon. Such function macros
+# are typically used for boiler-plate code, and will confuse the parser if not
+# removed.
+# The default value is: YES.
+# This tag requires that the tag ENABLE_PREPROCESSING is set to YES.
+
+SKIP_FUNCTION_MACROS = YES
+
+#---------------------------------------------------------------------------
+# Configuration options related to external references
+#---------------------------------------------------------------------------
+
+# The TAGFILES tag can be used to specify one or more tag files. For each tag
+# file the location of the external documentation should be added. The format of
+# a tag file without this location is as follows:
+# TAGFILES = file1 file2 ...
+# Adding location for the tag files is done as follows:
+# TAGFILES = file1=loc1 "file2 = loc2" ...
+# where loc1 and loc2 can be relative or absolute paths or URLs. See the
+# section "Linking to external documentation" for more information about the use
+# of tag files.
+# Note: Each tag file must have a unique name (where the name does NOT include
+# the path). If a tag file is not located in the directory in which doxygen is
+# run, you must also specify the path to the tagfile here.
+
+TAGFILES =
+
+# When a file name is specified after GENERATE_TAGFILE, doxygen will create a
+# tag file that is based on the input files it reads. See section "Linking to
+# external documentation" for more information about the usage of tag files.
+
+GENERATE_TAGFILE =
+
+# If the ALLEXTERNALS tag is set to YES, all external class will be listed in
+# the class index. If set to NO, only the inherited external classes will be
+# listed.
+# The default value is: NO.
+
+ALLEXTERNALS = NO
+
+# If the EXTERNAL_GROUPS tag is set to YES, all external groups will be listed
+# in the modules index. If set to NO, only the current project's groups will be
+# listed.
+# The default value is: YES.
+
+EXTERNAL_GROUPS = YES
+
+# If the EXTERNAL_PAGES tag is set to YES, all external pages will be listed in
+# the related pages index. If set to NO, only the current project's pages will
+# be listed.
+# The default value is: YES.
+
+EXTERNAL_PAGES = YES
+
+# The PERL_PATH should be the absolute path and name of the perl script
+# interpreter (i.e. the result of 'which perl').
+# The default file (with absolute path) is: /usr/bin/perl.
+
+PERL_PATH = /usr/bin/perl
+
+#---------------------------------------------------------------------------
+# Configuration options related to the dot tool
+#---------------------------------------------------------------------------
+
+# If the CLASS_DIAGRAMS tag is set to YES, doxygen will generate a class diagram
+# (in HTML and LaTeX) for classes with base or super classes. Setting the tag to
+# NO turns the diagrams off. Note that this option also works with HAVE_DOT
+# disabled, but it is recommended to install and use dot, since it yields more
+# powerful graphs.
+# The default value is: YES.
+
+CLASS_DIAGRAMS = NO
+
+# You can define message sequence charts within doxygen comments using the \msc
+# command. Doxygen will then run the mscgen tool (see:
+# http://www.mcternan.me.uk/mscgen/)) to produce the chart and insert it in the
+# documentation. The MSCGEN_PATH tag allows you to specify the directory where
+# the mscgen tool resides. If left empty the tool is assumed to be found in the
+# default search path.
+
+MSCGEN_PATH =
+
+# You can include diagrams made with dia in doxygen documentation. Doxygen will
+# then run dia to produce the diagram and insert it in the documentation. The
+# DIA_PATH tag allows you to specify the directory where the dia binary resides.
+# If left empty dia is assumed to be found in the default search path.
+
+DIA_PATH =
+
+# If set to YES the inheritance and collaboration graphs will hide inheritance
+# and usage relations if the target is undocumented or is not a class.
+# The default value is: YES.
+
+HIDE_UNDOC_RELATIONS = YES
+
+# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is
+# available from the path. This tool is part of Graphviz (see:
+# http://www.graphviz.org/), a graph visualization toolkit from AT&T and Lucent
+# Bell Labs. The other options in this section have no effect if this option is
+# set to NO
+# The default value is: NO.
+
+HAVE_DOT = NO
+
+# The DOT_NUM_THREADS specifies the number of dot invocations doxygen is allowed
+# to run in parallel. When set to 0 doxygen will base this on the number of
+# processors available in the system. You can set it explicitly to a value
+# larger than 0 to get control over the balance between CPU load and processing
+# speed.
+# Minimum value: 0, maximum value: 32, default value: 0.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_NUM_THREADS = 0
+
+# When you want a differently looking font in the dot files that doxygen
+# generates you can specify the font name using DOT_FONTNAME. You need to make
+# sure dot is able to find the font, which can be done by putting it in a
+# standard location or by setting the DOTFONTPATH environment variable or by
+# setting DOT_FONTPATH to the directory containing the font.
+# The default value is: Helvetica.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_FONTNAME =
+
+# The DOT_FONTSIZE tag can be used to set the size (in points) of the font of
+# dot graphs.
+# Minimum value: 4, maximum value: 24, default value: 10.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_FONTSIZE = 10
+
+# By default doxygen will tell dot to use the default font as specified with
+# DOT_FONTNAME. If you specify a different font using DOT_FONTNAME you can set
+# the path where dot can find it using this tag.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_FONTPATH =
+
+# If the CLASS_GRAPH tag is set to YES then doxygen will generate a graph for
+# each documented class showing the direct and indirect inheritance relations.
+# Setting this tag to YES will force the CLASS_DIAGRAMS tag to NO.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+CLASS_GRAPH = NO
+
+# If the COLLABORATION_GRAPH tag is set to YES then doxygen will generate a
+# graph for each documented class showing the direct and indirect implementation
+# dependencies (inheritance, containment, and class references variables) of the
+# class with other documented classes.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+COLLABORATION_GRAPH = NO
+
+# If the GROUP_GRAPHS tag is set to YES then doxygen will generate a graph for
+# groups, showing the direct groups dependencies.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+GROUP_GRAPHS = YES
+
+# If the UML_LOOK tag is set to YES, doxygen will generate inheritance and
+# collaboration diagrams in a style similar to the OMG's Unified Modeling
+# Language.
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+UML_LOOK = NO
+
+# If the UML_LOOK tag is enabled, the fields and methods are shown inside the
+# class node. If there are many fields or methods and many nodes the graph may
+# become too big to be useful. The UML_LIMIT_NUM_FIELDS threshold limits the
+# number of items for each type to make the size more manageable. Set this to 0
+# for no limit. Note that the threshold may be exceeded by 50% before the limit
+# is enforced. So when you set the threshold to 10, up to 15 fields may appear,
+# but if the number exceeds 15, the total amount of fields shown is limited to
+# 10.
+# Minimum value: 0, maximum value: 100, default value: 10.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+UML_LIMIT_NUM_FIELDS = 10
+
+# If the TEMPLATE_RELATIONS tag is set to YES then the inheritance and
+# collaboration graphs will show the relations between templates and their
+# instances.
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+TEMPLATE_RELATIONS = NO
+
+# If the INCLUDE_GRAPH, ENABLE_PREPROCESSING and SEARCH_INCLUDES tags are set to
+# YES then doxygen will generate a graph for each documented file showing the
+# direct and indirect include dependencies of the file with other documented
+# files.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+INCLUDE_GRAPH = NO
+
+# If the INCLUDED_BY_GRAPH, ENABLE_PREPROCESSING and SEARCH_INCLUDES tags are
+# set to YES then doxygen will generate a graph for each documented file showing
+# the direct and indirect include dependencies of the file with other documented
+# files.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+INCLUDED_BY_GRAPH = NO
+
+# If the CALL_GRAPH tag is set to YES then doxygen will generate a call
+# dependency graph for every global function or class method.
+#
+# Note that enabling this option will significantly increase the time of a run.
+# So in most cases it will be better to enable call graphs for selected
+# functions only using the \callgraph command. Disabling a call graph can be
+# accomplished by means of the command \hidecallgraph.
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+CALL_GRAPH = NO
+
+# If the CALLER_GRAPH tag is set to YES then doxygen will generate a caller
+# dependency graph for every global function or class method.
+#
+# Note that enabling this option will significantly increase the time of a run.
+# So in most cases it will be better to enable caller graphs for selected
+# functions only using the \callergraph command. Disabling a caller graph can be
+# accomplished by means of the command \hidecallergraph.
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+CALLER_GRAPH = NO
+
+# If the GRAPHICAL_HIERARCHY tag is set to YES then doxygen will graphical
+# hierarchy of all classes instead of a textual one.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+GRAPHICAL_HIERARCHY = NO
+
+# If the DIRECTORY_GRAPH tag is set to YES then doxygen will show the
+# dependencies a directory has on other directories in a graphical way. The
+# dependency relations are determined by the #include relations between the
+# files in the directories.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DIRECTORY_GRAPH = NO
+
+# The DOT_IMAGE_FORMAT tag can be used to set the image format of the images
+# generated by dot. For an explanation of the image formats see the section
+# output formats in the documentation of the dot tool (Graphviz (see:
+# http://www.graphviz.org/)).
+# Note: If you choose svg you need to set HTML_FILE_EXTENSION to xhtml in order
+# to make the SVG files visible in IE 9+ (other browsers do not have this
+# requirement).
+# Possible values are: png, jpg, gif, svg, png:gd, png:gd:gd, png:cairo,
+# png:cairo:gd, png:cairo:cairo, png:cairo:gdiplus, png:gdiplus and
+# png:gdiplus:gdiplus.
+# The default value is: png.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_IMAGE_FORMAT = png
+
+# If DOT_IMAGE_FORMAT is set to svg, then this option can be set to YES to
+# enable generation of interactive SVG images that allow zooming and panning.
+#
+# Note that this requires a modern browser other than Internet Explorer. Tested
+# and working are Firefox, Chrome, Safari, and Opera.
+# Note: For IE 9+ you need to set HTML_FILE_EXTENSION to xhtml in order to make
+# the SVG files visible. Older versions of IE do not have SVG support.
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+INTERACTIVE_SVG = NO
+
+# The DOT_PATH tag can be used to specify the path where the dot tool can be
+# found. If left blank, it is assumed the dot tool can be found in the path.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_PATH =
+
+# The DOTFILE_DIRS tag can be used to specify one or more directories that
+# contain dot files that are included in the documentation (see the \dotfile
+# command).
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOTFILE_DIRS =
+
+# The MSCFILE_DIRS tag can be used to specify one or more directories that
+# contain msc files that are included in the documentation (see the \mscfile
+# command).
+
+MSCFILE_DIRS =
+
+# The DIAFILE_DIRS tag can be used to specify one or more directories that
+# contain dia files that are included in the documentation (see the \diafile
+# command).
+
+DIAFILE_DIRS =
+
+# When using plantuml, the PLANTUML_JAR_PATH tag should be used to specify the
+# path where java can find the plantuml.jar file. If left blank, it is assumed
+# PlantUML is not used or called during a preprocessing step. Doxygen will
+# generate a warning when it encounters a \startuml command in this case and
+# will not generate output for the diagram.
+
+PLANTUML_JAR_PATH =
+
+# When using plantuml, the specified paths are searched for files specified by
+# the !include statement in a plantuml block.
+
+PLANTUML_INCLUDE_PATH =
+
+# The DOT_GRAPH_MAX_NODES tag can be used to set the maximum number of nodes
+# that will be shown in the graph. If the number of nodes in a graph becomes
+# larger than this value, doxygen will truncate the graph, which is visualized
+# by representing a node as a red box. Note that doxygen if the number of direct
+# children of the root node in a graph is already larger than
+# DOT_GRAPH_MAX_NODES then the graph will not be shown at all. Also note that
+# the size of a graph can be further restricted by MAX_DOT_GRAPH_DEPTH.
+# Minimum value: 0, maximum value: 10000, default value: 50.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_GRAPH_MAX_NODES = 50
+
+# The MAX_DOT_GRAPH_DEPTH tag can be used to set the maximum depth of the graphs
+# generated by dot. A depth value of 3 means that only nodes reachable from the
+# root by following a path via at most 3 edges will be shown. Nodes that lay
+# further from the root node will be omitted. Note that setting this option to 1
+# or 2 may greatly reduce the computation time needed for large code bases. Also
+# note that the size of a graph can be further restricted by
+# DOT_GRAPH_MAX_NODES. Using a depth of 0 means no depth restriction.
+# Minimum value: 0, maximum value: 1000, default value: 0.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+MAX_DOT_GRAPH_DEPTH = 0
+
+# Set the DOT_TRANSPARENT tag to YES to generate images with a transparent
+# background. This is disabled by default, because dot on Windows does not seem
+# to support this out of the box.
+#
+# Warning: Depending on the platform used, enabling this option may lead to
+# badly anti-aliased labels on the edges of a graph (i.e. they become hard to
+# read).
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_TRANSPARENT = NO
+
+# Set the DOT_MULTI_TARGETS tag to YES to allow dot to generate multiple output
+# files in one run (i.e. multiple -o and -T options on the command line). This
+# makes dot run faster, but since only newer versions of dot (>1.8.10) support
+# this, this feature is disabled by default.
+# The default value is: NO.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_MULTI_TARGETS = NO
+
+# If the GENERATE_LEGEND tag is set to YES doxygen will generate a legend page
+# explaining the meaning of the various boxes and arrows in the dot generated
+# graphs.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+GENERATE_LEGEND = NO
+
+# If the DOT_CLEANUP tag is set to YES, doxygen will remove the intermediate dot
+# files that are used to generate the various graphs.
+# The default value is: YES.
+# This tag requires that the tag HAVE_DOT is set to YES.
+
+DOT_CLEANUP = YES
diff --git a/doc/style.css b/doc/style.css
new file mode 100644
index 0000000..f6ff8bb
--- /dev/null
+++ b/doc/style.css
@@ -0,0 +1,691 @@
+body {
+ max-width: 80em;
+ margin: 0;
+ margin-left: auto;
+ margin-right: auto;
+ background: #FFF;
+ color: #000;
+}
+
+#titlearea {
+ display: none;
+}
+
+h1 {
+ font-size: 180%;
+ font-weight: 900;
+}
+
+h2 {
+ font-size: 140%;
+ font-weight: 700;
+}
+
+h3 {
+ font-size: 120%;
+ font-weight: 700;
+}
+
+h4 {
+ font-size: 110%;
+ font-weight: 700;
+}
+
+h5 {
+ font-size: 100%;
+ font-weight: 700;
+}
+
+h6 {
+ font-size: 100%;
+ font-weight: 600;
+}
+
+p {
+ margin: 0 0 1em 0;
+}
+
+dt {
+ font-weight: 700;
+}
+
+p.startli,p.startdd,p.starttd {
+ margin-top: 2px;
+}
+
+p.endli {
+ margin-bottom: 0;
+}
+
+p.enddd {
+ margin-bottom: 4px;
+}
+
+p.endtd {
+ margin-bottom: 2px;
+}
+
+caption {
+ font-weight: 700;
+}
+
+span.legend {
+ font-size: small;
+ text-align: center;
+}
+
+h3.version {
+ font-size: small;
+ text-align: center;
+}
+
+div.qindex,div.navtab {
+ background-color: #EBEFF6;
+ border: 1px solid #A3B4D7;
+ text-align: center;
+ margin: 2px;
+ padding: 2px;
+}
+
+div.navtab {
+ margin-right: 15px;
+}
+
+/* @group Link Styling */
+a {
+ color: #546E00;
+ text-decoration: none;
+}
+
+.contents a:visited {
+ color: #344E00;
+}
+
+a:hover {
+ text-decoration: underline;
+}
+
+a.qindexHL {
+ background-color: #9CAFD4;
+ color: #FFF;
+ border: 1px double #869DCA;
+}
+
+code {
+ color: #444;
+}
+
+a.code {
+ color: #4665A2;
+}
+
+a.codeRef {
+ color: #4665A2;
+}
+
+/* @end */
+dl.el {
+ margin-left: -1cm;
+}
+
+.fragment {
+ font-family: monospace, fixed;
+}
+
+pre.fragment {
+ border: 1px solid #C4C4C4;
+ background-color: #F9F9F9;
+ padding: 4px 6px;
+ margin: 4px 8px 4px 2px;
+ overflow: auto;
+ line-height: 125%;
+}
+
+div.ah {
+ background-color: #000;
+ font-weight: 700;
+ color: #FFF;
+ margin-bottom: 3px;
+ margin-top: 3px;
+ padding: .2em;
+ border: thin solid #333;
+}
+
+div.groupHeader {
+ margin-left: 16px;
+ margin-top: 12px;
+ margin-bottom: 6px;
+ font-weight: 700;
+}
+
+a + h2.groupheader {
+ display: none;
+}
+
+div.groupText {
+ margin-left: 16px;
+ font-style: italic;
+}
+
+div.contents {
+ margin-top: 10px;
+ margin-left: 10px;
+ margin-right: 10px;
+}
+
+td.indexkey {
+ background-color: #EBEFF6;
+ font-weight: 700;
+ border: 1px solid #C4CFE5;
+ margin: 2px 0;
+ padding: 2px 10px;
+}
+
+td.indexvalue {
+ background-color: #EBEFF6;
+ border: 1px solid #C4CFE5;
+ padding: 2px 10px;
+ margin: 2px 0;
+}
+
+tr.memlist {
+ background-color: #EEF1F7;
+}
+
+p.formulaDsp {
+ text-align: center;
+}
+
+img.formulaInl {
+ vertical-align: middle;
+}
+
+div.center {
+ text-align: center;
+ margin-top: 0;
+ margin-bottom: 0;
+ padding: 0;
+}
+
+div.center img {
+ border: 0;
+}
+
+address.footer {
+ text-align: right;
+ padding-right: 12px;
+}
+
+img.footer {
+ border: 0;
+ vertical-align: middle;
+}
+
+/* @group Code Colorization */
+span.keyword {
+ color: green;
+}
+
+span.keywordtype {
+ color: #3E873E;
+}
+
+span.keywordflow {
+ color: #e08000;
+}
+
+span.comment {
+ color: maroon;
+}
+
+span.preprocessor {
+ color: #806020;
+}
+
+span.stringliteral {
+ color: #002080;
+}
+
+span.charliteral {
+ color: teal;
+}
+
+span.vhdldigit {
+ color: #F0F;
+}
+
+span.vhdlkeyword {
+ color: #700070;
+}
+
+span.vhdllogic {
+ color: red;
+}
+
+/* @end */
+td.tiny {
+ font-size: x-small;
+}
+
+.dirtab {
+ padding: 4px;
+ border-collapse: collapse;
+ border: 1px solid #A3B4D7;
+}
+
+th.dirtab {
+ background: #EBEFF6;
+ font-weight: 700;
+}
+
+hr {
+ height: 0;
+ border: none;
+ border-top: 1px solid #DDD;
+ margin: 2em 0 1em;
+}
+
+hr.footer {
+ height: 1px;
+}
+
+/* @group Member Descriptions */
+table.memberdecls {
+ border-spacing: 0.125em;
+}
+
+h2.groupheader {
+ margin: 0.5em 0 0.25em 0;
+}
+
+.mdescLeft,.mdescRight,.memItemLeft,.memItemRight,.memTemplItemLeft,.memTemplItemRight,.memTemplParams {
+ margin: 0;
+ padding: 0;
+}
+
+.mdescLeft,.mdescRight {
+ color: #555;
+}
+
+.memItemLeft,.memItemRight,.memTemplParams {
+ border: 0;
+ font-family: monospace, fixed;
+}
+
+.memItemLeft,.memTemplItemLeft {
+ white-space: nowrap;
+ padding-left: 2em;
+ padding-right: 1em;
+}
+
+.memItemLeft a.el {
+ font-weight: bold;
+}
+
+.memTemplParams {
+ color: #464646;
+ white-space: nowrap;
+}
+
+td.memSeparator {
+ display: none;
+}
+
+td.mlabels-right {
+ vertical-align: top;
+ padding-top: 4px;
+ color: #AA6;
+}
+
+.memtitle {
+ display: none;
+}
+
+/* @end */
+/* @group Member Details */
+/* Styles for detailed member documentation */
+.memtemplate {
+ color: #4665A2;
+ font-weight: bold;
+}
+
+.memnav {
+ background-color: #EBEFF6;
+ border: 1px solid #A3B4D7;
+ text-align: center;
+ margin: 2px;
+ margin-right: 15px;
+ padding: 2px;
+}
+
+.memitem {
+ padding: 0;
+ margin: 1em 0 1em 0;
+}
+
+.memproto {
+ padding: 0;
+ font-weight: bold;
+ color: #000;
+}
+
+.memproto .paramname {
+ color: #444;
+ font-style: normal;
+}
+
+.memdoc {
+ padding: 0 0 0.5em 2em;
+}
+
+.paramkey {
+ text-align: right;
+}
+
+.paramtype {
+ color: #3E873E;
+ white-space: nowrap;
+}
+
+.paramname {
+ color: #444;
+ white-space: nowrap;
+ font-weight: bold;
+}
+
+td.paramname {
+ vertical-align: top;
+}
+
+.fieldname {
+ color: #000;
+}
+
+td.fieldname {
+ padding-right: 1em;
+ vertical-align: top;
+}
+
+td.fieldtype {
+ vertical-align: top;
+ color: #444;
+ padding-right: 0.5em;
+}
+
+td.fielddoc p {
+ margin: 0;
+}
+
+/* @end */
+/* @group Directory (tree) */
+/* for the tree view */
+.ftvtree {
+ font-family: sans-serif;
+ margin: 0;
+}
+
+/* these are for tree view when used as main index */
+.directory {
+ font-size: small;
+ margin: 0.5em;
+}
+
+.directory h3 {
+ margin: 0;
+ margin-top: 1em;
+ font-size: 11pt;
+}
+
+.directory > h3 {
+ margin-top: 0;
+}
+
+.directory p {
+ margin: 0;
+ white-space: nowrap;
+}
+
+.directory div {
+ display: none;
+ margin: 0;
+}
+
+.directory img {
+ vertical-align: -30%;
+}
+
+/* these are for tree view when not used as main index */
+.directory-alt {
+ font-size: 100%;
+ font-weight: bold;
+}
+
+.directory-alt h3 {
+ margin: 0;
+ margin-top: 1em;
+ font-size: 11pt;
+}
+
+.directory-alt > h3 {
+ margin-top: 0;
+}
+
+.directory-alt p {
+ margin: 0;
+ white-space: nowrap;
+}
+
+.directory-alt div {
+ display: none;
+ margin: 0;
+}
+
+.directory-alt img {
+ vertical-align: -30%;
+}
+
+/* @end */
+div.dynheader {
+ margin-top: 8px;
+}
+
+address {
+ font-style: normal;
+ color: #2A3D61;
+}
+
+table.doxtable {
+ border-collapse: collapse;
+ margin: 0.5em;
+}
+
+table.doxtable td,table.doxtable th {
+ border: 1px solid #DDD;
+ padding: 3px 7px 2px;
+}
+
+table.doxtable th {
+ background-color: #F3F3F3;
+ color: #000;
+ padding-bottom: 4px;
+ padding-top: 5px;
+ text-align: left;
+ font-weight: bold;
+}
+
+.tabsearch {
+ top: 0;
+ left: 10px;
+ height: 36px;
+ z-index: 101;
+ overflow: hidden;
+ font-size: 13px;
+}
+
+div.navpath {
+ padding: 0.25em;
+}
+
+.navpath ul {
+ font-size: x-small;
+ color: #8AA0CC;
+ overflow: hidden;
+ margin: 0;
+ padding: 0;
+}
+
+.navpath li {
+ list-style-type: none;
+ float: left;
+ padding-left: 10px;
+ padding-right: 15px;
+ color: #364D7C;
+}
+
+.navpath a {
+ display: block;
+ text-decoration: none;
+ outline: none;
+}
+
+.navpath a:hover {
+ color: #6884BD;
+}
+
+div.summary {
+ float: right;
+ font-size: x-small;
+ padding: 0.25em 0.5em 0 0;
+ width: 50%;
+ text-align: right;
+}
+
+div.summary a {
+ white-space: nowrap;
+}
+
+div.header {
+ background-color: #F3F3F3;
+ margin: 0;
+ border: 0;
+}
+
+div.headertitle {
+ font-size: 180%;
+ font-weight: bold;
+ color: #FFF;
+ padding: 0.125em 0.25em 0.125em 0.25em;
+ background-color: #333;
+ background: linear-gradient(to bottom, #333 0%, #111 100%);
+ border: solid 1px #444;
+ border-top: 0;
+ border-radius: 0 0 6px 6px;
+}
+
+div.line {
+ font-family: monospace, fixed;
+ font-size: 13px;
+ min-height: 13px;
+ line-height: 1.0;
+ text-wrap: avoid;
+ white-space: pre-wrap;
+ text-indent: -53px;
+ padding-left: 53px;
+ padding-bottom: 0;
+ margin: 0;
+}
+
+.glow {
+ background-color: cyan;
+ box-shadow: 0 0 10px cyan;
+}
+
+span.lineno {
+ padding-right: 4px;
+ text-align: right;
+ border-right: 2px solid #0F0;
+ background-color: #E8E8E8;
+ white-space: pre;
+}
+span.lineno a {
+ background-color: #D8D8D8;
+}
+
+span.lineno a:hover {
+ background-color: #C8C8C8;
+}
+
+.tabs, .tabs2, .navpath {
+ background-image: none;
+ background-color: #333;
+ background: linear-gradient(to bottom, #333 0%, #111 100%);
+ border: 0;
+ border-bottom: solid 2px #000;
+ padding: 0;
+ padding-top: 2px;
+ font-size: small;
+}
+
+#navrow1 {
+ border: 0;
+}
+
+th {
+ text-align: left;
+}
+
+.mlabel {
+ padding: 0.125em;
+}
+
+/* tabs*/
+
+.tablist {
+ margin: 0;
+ padding: 0;
+ display: table;
+}
+
+.tablist li {
+ display: table-cell;
+ line-height: 2em;
+ list-style: none;
+ background-color: #333;
+ background: linear-gradient(to bottom, #444 0%, #222 100%);
+ border: 1px solid #222;
+ border-bottom: 0;
+ border-radius: 6px 6px 0 0;
+ color: #DDD;
+}
+
+.tablist a {
+ display: block;
+ padding: 0 20px;
+ font-weight: bold;
+ color: #859900;
+ text-decoration: none;
+ outline: none;
+}
+
+.header a {
+ color: #859900;
+}
+
+.tabs3 .tablist a {
+ padding: 0 10px;
+}
+
+.tablist a:hover {
+ color: #fff;
+ text-shadow: 0 1px 1px rgba(0, 0, 0, 1.0);
+ text-decoration: none;
+}
+
+.tablist li.current a {
+ color: #fff;
+ text-shadow: 0 1px 1px rgba(0, 0, 0, 1.0);
+}
+
+span.icon {
+ display: none;
+}
diff --git a/src/cocoa_in_gtk2.mm b/src/cocoa_in_gtk2.mm
new file mode 100644
index 0000000..3ff87d0
--- /dev/null
+++ b/src/cocoa_in_gtk2.mm
@@ -0,0 +1,433 @@
+/*
+ Copyright 2011-2017 David Robillard <http://drobilla.net>
+ Copyright 2014 Robin Gareus <robin@gareus.org>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <string.h>
+
+#include <gtk/gtk.h>
+#include <gdk/gdkquartz.h>
+
+#include "./suil_internal.h"
+
+#include "lv2/lv2plug.in/ns/ext/options/options.h"
+#include "lv2/lv2plug.in/ns/ext/urid/urid.h"
+
+#if MAC_OS_X_VERSION_MAX_ALLOWED < MAC_OS_X_VERSION_10_12
+#define NSEventTypeFlagsChanged NSFlagsChanged
+#define NSEventTypeLeftMouseDown NSLeftMouseDown
+#define NSEventTypeLeftMouseDragged NSLeftMouseDragged
+#define NSEventTypeLeftMouseUp NSLeftMouseUp
+#define NSEventTypeMouseEntered NSMouseEntered
+#define NSEventTypeMouseExited NSMouseExited
+#define NSEventTypeMouseMoved NSMouseMoved
+#define NSEventTypeRightMouseDown NSRightMouseDown
+#define NSEventTypeRightMouseUp NSRightMouseUp
+#define NSEventTypeScrollWheel NSScrollWheel
+#endif
+
+extern "C" {
+
+#define SUIL_TYPE_COCOA_WRAPPER (suil_cocoa_wrapper_get_type())
+#define SUIL_COCOA_WRAPPER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SUIL_TYPE_COCOA_WRAPPER, SuilCocoaWrapper))
+
+typedef struct _SuilCocoaWrapper SuilCocoaWrapper;
+typedef struct _SuilCocoaWrapperClass SuilCocoaWrapperClass;
+
+struct _SuilCocoaWrapper {
+ GtkWidget widget;
+ SuilWrapper* wrapper;
+ SuilInstance* instance;
+
+ GdkWindow* flt_win;
+ bool custom_size;
+ bool mapped;
+ int req_width;
+ int req_height;
+ int alo_width;
+ int alo_height;
+
+ const LV2UI_Idle_Interface* idle_iface;
+ guint idle_id;
+ guint idle_ms;
+};
+
+struct _SuilCocoaWrapperClass {
+ GtkWidgetClass parent_class;
+};
+
+GType suil_cocoa_wrapper_get_type(void); // Accessor for SUIL_TYPE_COCOA_WRAPPER
+
+G_DEFINE_TYPE(SuilCocoaWrapper, suil_cocoa_wrapper, GTK_TYPE_WIDGET)
+
+static void
+suil_cocoa_wrapper_finalize(GObject* gobject)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(gobject);
+
+ self->wrapper->impl = NULL;
+
+ G_OBJECT_CLASS(suil_cocoa_wrapper_parent_class)->finalize(gobject);
+}
+
+static void
+suil_cocoa_realize(GtkWidget* widget)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ g_return_if_fail(self != NULL);
+
+ GTK_WIDGET_SET_FLAGS(widget, GTK_REALIZED);
+
+ GdkWindowAttr attrs;
+ attrs.x = widget->allocation.x;
+ attrs.y = widget->allocation.y;
+ attrs.width = widget->allocation.width;
+ attrs.height = widget->allocation.height;
+ attrs.wclass = GDK_INPUT_OUTPUT;
+ attrs.window_type = GDK_WINDOW_CHILD;
+ attrs.visual = gtk_widget_get_visual(widget);
+ attrs.colormap = gtk_widget_get_colormap(widget);
+ attrs.event_mask = gtk_widget_get_events(widget) |
+ GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK |
+ GDK_BUTTON_RELEASE_MASK | GDK_POINTER_MOTION_MASK |
+ GDK_POINTER_MOTION_HINT_MASK;
+
+ widget->window = gdk_window_new(
+ widget->parent->window,
+ &attrs,
+ GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP);
+
+ widget->style = gtk_style_attach(widget->style, widget->window);
+
+ gdk_window_set_user_data(widget->window, widget);
+ gtk_style_set_background(widget->style, widget->window, GTK_STATE_ACTIVE);
+ gtk_widget_queue_resize(widget);
+}
+
+static void
+suil_cocoa_size_request(GtkWidget* widget, GtkRequisition* requisition)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ if (self->custom_size) {
+ requisition->width = self->req_width;
+ requisition->height = self->req_height;
+ } else {
+ NSView* view = (NSView*)self->instance->ui_widget;
+ NSRect frame = [view frame];
+ requisition->width = CGRectGetWidth(NSRectToCGRect(frame));
+ requisition->height = CGRectGetHeight(NSRectToCGRect(frame));
+ }
+}
+
+static void
+suil_cocoa_size_allocate(GtkWidget* widget, GtkAllocation* allocation)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ self->alo_width = allocation->width;
+ self->alo_height = allocation->height;
+
+ if (!self->mapped) {
+ return;
+ }
+
+ gint xx, yy;
+ gtk_widget_translate_coordinates(
+ gtk_widget_get_parent(widget), widget, 0, 0, &xx, &yy);
+
+ NSView* view = (NSView*)self->instance->ui_widget;
+ [view setFrame:NSMakeRect(xx, yy, self->alo_width, self->alo_height)];
+}
+
+static void
+suil_cocoa_map(GtkWidget* widget)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ self->mapped = true;
+
+ if (self->alo_width == 0 || self->alo_height ==0) {
+ return;
+ }
+
+ gint xx, yy;
+ gtk_widget_translate_coordinates(
+ gtk_widget_get_parent(widget), widget, 0, 0, &xx, &yy);
+
+ NSView* view = (NSView*)self->instance->ui_widget;
+ [view setHidden:NO];
+ [view setFrame:NSMakeRect(xx, yy, self->alo_width, self->alo_height)];
+}
+
+static void
+suil_cocoa_unmap(GtkWidget* widget)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ NSView* view = (NSView*)self->instance->ui_widget;
+
+ self->mapped = false;
+ [view setHidden:YES];
+}
+
+static gboolean
+suil_cocoa_key_press(GtkWidget* widget, GdkEventKey* event)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ if (!self->instance || !self->wrapper || !self->wrapper->impl) {
+ return FALSE;
+ }
+ NSEvent* nsevent = gdk_quartz_event_get_nsevent((GdkEvent*)event);
+ NSView* view = (NSView*)self->instance->ui_widget;
+ [view keyDown:nsevent];
+ return TRUE;
+}
+
+static gboolean
+suil_cocoa_key_release(GtkWidget* widget, GdkEventKey* event)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ if (!self->instance || !self->wrapper || !self->wrapper->impl) {
+ return FALSE;
+ }
+ NSEvent* nsevent = gdk_quartz_event_get_nsevent((GdkEvent*)event);
+ NSView* view = (NSView*)self->instance->ui_widget;
+ [view keyUp:nsevent];
+ return TRUE;
+}
+
+static gboolean
+suil_cocoa_expose(GtkWidget* widget, GdkEventExpose* event)
+{
+ SuilCocoaWrapper* const self = SUIL_COCOA_WRAPPER(widget);
+ NSView* view = (NSView*)self->instance->ui_widget;
+ [view drawRect:NSMakeRect(event->area.x,
+ event->area.y,
+ event->area.width,
+ event->area.height)];
+ return TRUE;
+}
+
+static void
+suil_cocoa_wrapper_class_init(SuilCocoaWrapperClass* klass)
+{
+ GObjectClass* const gobject_class = G_OBJECT_CLASS(klass);
+ GtkWidgetClass* const widget_class = (GtkWidgetClass*)(klass);
+
+ gobject_class->finalize = suil_cocoa_wrapper_finalize;
+
+ widget_class->realize = suil_cocoa_realize;
+ widget_class->expose_event = suil_cocoa_expose;
+ widget_class->size_request = suil_cocoa_size_request;
+ widget_class->size_allocate = suil_cocoa_size_allocate;
+ widget_class->map = suil_cocoa_map;
+ widget_class->unmap = suil_cocoa_unmap;
+ widget_class->key_press_event = suil_cocoa_key_press;
+ widget_class->key_release_event = suil_cocoa_key_release;
+}
+
+static void
+suil_cocoa_wrapper_init(SuilCocoaWrapper* self)
+{
+ self->wrapper = NULL;
+ self->instance = NULL;
+ self->flt_win = NULL;
+ self->custom_size = false;
+ self->mapped = false;
+ self->req_width = self->req_height = 0;
+ self->alo_width = self->alo_height = 0;
+ self->idle_iface = NULL;
+ self->idle_ms = 1000 / 30; // 30 Hz default
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ SuilCocoaWrapper* const wrap = SUIL_COCOA_WRAPPER(handle);
+ wrap->req_width = width;
+ wrap->req_height = height;
+ wrap->custom_size = true;
+ gtk_widget_queue_resize(GTK_WIDGET(handle));
+ return 0;
+}
+
+static gboolean
+suil_cocoa_wrapper_idle(void* data)
+{
+ SuilCocoaWrapper* const wrap = SUIL_COCOA_WRAPPER(data);
+ wrap->idle_iface->idle(wrap->instance->handle);
+ return TRUE; // Continue calling
+}
+
+static GdkFilterReturn
+event_filter(GdkXEvent* xevent, GdkEvent* event, gpointer data)
+{
+ SuilCocoaWrapper* wrap = (SuilCocoaWrapper*)data;
+ if (!wrap->instance || !wrap->wrapper || !wrap->wrapper->impl) {
+ return GDK_FILTER_CONTINUE;
+ }
+
+ NSEvent* nsevent = (NSEvent*)xevent;
+ NSView* view = (NSView*)wrap->instance->ui_widget;
+ if (view && nsevent) {
+ switch([nsevent type]) {
+ case NSEventTypeFlagsChanged:
+ [view flagsChanged:nsevent];
+ return GDK_FILTER_REMOVE;
+ case NSEventTypeMouseEntered:
+ [view mouseEntered:nsevent];
+ return GDK_FILTER_REMOVE;
+ case NSEventTypeMouseExited:
+ [view mouseExited:nsevent];
+ return GDK_FILTER_REMOVE;
+
+ /* Explicitly pass though mouse events. Needed for mouse-drags leaving
+ the window, and mouse-up after that. */
+ case NSEventTypeMouseMoved:
+ [view mouseMoved:nsevent];
+ break;
+ case NSEventTypeLeftMouseDragged:
+ [view mouseDragged:nsevent];
+ break;
+ case NSEventTypeLeftMouseDown:
+ [view mouseDown:nsevent];
+ break;
+ case NSEventTypeLeftMouseUp:
+ [view mouseUp:nsevent];
+ break;
+ case NSEventTypeRightMouseDown:
+ [view rightMouseDown:nsevent];
+ break;
+ case NSEventTypeRightMouseUp:
+ [view rightMouseUp:nsevent];
+ break;
+ case NSEventTypeScrollWheel:
+ [view scrollWheel:nsevent];
+ break;
+ default:
+ break;
+ }
+ }
+ return GDK_FILTER_CONTINUE;
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper, SuilInstance* instance)
+{
+ SuilCocoaWrapper* const wrap = SUIL_COCOA_WRAPPER(wrapper->impl);
+
+ instance->host_widget = GTK_WIDGET(wrap);
+ wrap->wrapper = wrapper;
+ wrap->instance = instance;
+
+ const LV2UI_Idle_Interface* idle_iface = NULL;
+ if (instance->descriptor->extension_data) {
+ idle_iface = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ }
+ if (idle_iface) {
+ wrap->idle_iface = idle_iface;
+ wrap->idle_id = g_timeout_add(
+ wrap->idle_ms, suil_cocoa_wrapper_idle, wrap);
+ }
+
+ return 0;
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ if (wrapper->impl) {
+ SuilCocoaWrapper* const wrap = SUIL_COCOA_WRAPPER(wrapper->impl);
+ if (wrap->idle_id) {
+ g_source_remove(wrap->idle_id);
+ wrap->idle_id = 0;
+ }
+
+ gdk_window_remove_filter(wrap->flt_win, event_filter, wrapper->impl);
+ gtk_object_destroy(GTK_OBJECT(wrap));
+ }
+}
+
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ GtkWidget* parent = NULL;
+ for (unsigned i = 0; i < n_features; ++i) {
+ if (!strcmp((*features)[i]->URI, LV2_UI__parent)) {
+ parent = (GtkWidget*)(*features)[i]->data;
+ }
+ }
+
+ if (!GTK_CONTAINER(parent)) {
+ SUIL_ERRORF("No GtkContainer parent given for %s UI\n",
+ ui_type_uri);
+ return NULL;
+ }
+
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ SuilCocoaWrapper* const wrap = SUIL_COCOA_WRAPPER(
+ g_object_new(SUIL_TYPE_COCOA_WRAPPER, NULL));
+
+ wrapper->impl = wrap;
+ wrapper->resize.handle = wrap;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ gtk_container_add(GTK_CONTAINER(parent), GTK_WIDGET(wrap));
+ gtk_widget_set_can_focus(GTK_WIDGET(wrap), TRUE);
+ gtk_widget_set_sensitive(GTK_WIDGET(wrap), TRUE);
+ gtk_widget_realize(GTK_WIDGET(wrap));
+
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(wrap));
+ wrap->flt_win = gtk_widget_get_window(parent);
+ gdk_window_add_filter(wrap->flt_win, event_filter, wrap);
+
+ NSView* parent_view = gdk_quartz_window_get_nsview(window);
+ suil_add_feature(features, &n_features, LV2_UI__parent, parent_view);
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ // Scan for URID map and options
+ LV2_URID_Map* map = NULL;
+ LV2_Options_Option* options = NULL;
+ for (LV2_Feature** f = *features; *f && (!map || !options); ++f) {
+ if (!strcmp((*f)->URI, LV2_OPTIONS__options)) {
+ options = (LV2_Options_Option*)(*f)->data;
+ } else if (!strcmp((*f)->URI, LV2_URID__map)) {
+ map = (LV2_URID_Map*)(*f)->data;
+ }
+ }
+
+ if (map && options) {
+ // Set UI update rate if given
+ LV2_URID ui_updateRate = map->map(map->handle, LV2_UI__updateRate);
+ for (LV2_Options_Option* o = options; o->key; ++o) {
+ if (o->key == ui_updateRate) {
+ wrap->idle_ms = 1000.0f / *(const float*)o->value;
+ break;
+ }
+ }
+ }
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/cocoa_in_qt5.mm b/src/cocoa_in_qt5.mm
new file mode 100644
index 0000000..4259075
--- /dev/null
+++ b/src/cocoa_in_qt5.mm
@@ -0,0 +1,166 @@
+/*
+ Copyright 2017 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#import <Cocoa/Cocoa.h>
+
+#include <QCloseEvent>
+#include <QMacCocoaViewContainer>
+#include <QTimerEvent>
+#include <QWidget>
+
+#undef signals
+
+#include "./suil_config.h"
+#include "./suil_internal.h"
+
+extern "C" {
+
+typedef struct {
+ QWidget* host_widget;
+ QWidget* parent;
+} SuilCocoaInQt5Wrapper;
+
+class SuilQCocoaWidget : public QMacCocoaViewContainer
+{
+public:
+ SuilQCocoaWidget(NSView* view, QWidget* parent)
+ : QMacCocoaViewContainer(view, parent)
+ , _instance(NULL)
+ , _idle_iface(NULL)
+ , _ui_timer(0)
+ {
+ }
+
+ void start_idle(SuilInstance* instance,
+ const LV2UI_Idle_Interface* idle_iface) {
+ _instance = instance;
+ _idle_iface = idle_iface;
+ NSView* view = (NSView*)instance->ui_widget;
+ setCocoaView((NSView*)instance->ui_widget);
+ setMinimumWidth([view fittingSize].width);
+ setMinimumHeight([view fittingSize].height);
+ if (_idle_iface && _ui_timer == 0) {
+ _ui_timer = this->startTimer(30);
+ }
+ }
+
+protected:
+ void timerEvent(QTimerEvent* event) {
+ if (event->timerId() == _ui_timer && _idle_iface) {
+ _idle_iface->idle(_instance->handle);
+ }
+ QWidget::timerEvent(event);
+ }
+
+ void closeEvent(QCloseEvent* event) {
+ if (_ui_timer && _idle_iface) {
+ this->killTimer(_ui_timer);
+ _ui_timer = 0;
+ }
+ QWidget::closeEvent(event);
+ }
+
+private:
+ SuilInstance* _instance;
+ const LV2UI_Idle_Interface* _idle_iface;
+ int _ui_timer;
+};
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ SuilCocoaInQt5Wrapper* impl = (SuilCocoaInQt5Wrapper*)wrapper->impl;
+
+ if (impl->host_widget) {
+ delete impl->host_widget;
+ }
+
+ free(impl);
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilCocoaInQt5Wrapper* const impl = (SuilCocoaInQt5Wrapper*)wrapper->impl;
+ SuilQCocoaWidget* const ew = (SuilQCocoaWidget*)impl->parent;
+
+ if (instance->descriptor->extension_data) {
+ const LV2UI_Idle_Interface* idle_iface
+ = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ ew->start_idle(instance, idle_iface);
+ }
+
+ impl->host_widget = ew;
+
+ instance->host_widget = impl->host_widget;
+
+ return 0;
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ SuilQCocoaWidget* const ew = (SuilQCocoaWidget*)handle;
+ ew->resize(width, height);
+ return 0;
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ QWidget* parent = NULL;
+ for (unsigned i = 0; i < n_features; ++i) {
+ if (!strcmp((*features)[i]->URI, LV2_UI__parent)) {
+ parent = (QWidget*)(*features)[i]->data;
+ }
+ }
+
+ if (!parent) {
+ SUIL_ERRORF("No QWidget parent given for %s UI\n", ui_type_uri);
+ return NULL;
+ }
+
+ SuilCocoaInQt5Wrapper* const impl = (SuilCocoaInQt5Wrapper*)
+ calloc(1, sizeof(SuilCocoaInQt5Wrapper));
+
+ SuilWrapper* wrapper = (SuilWrapper*)malloc(sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+ NSView* view = [NSView new];
+
+ SuilQCocoaWidget* const ew = new SuilQCocoaWidget(view, parent);
+
+ impl->parent = ew;
+
+ wrapper->impl = impl;
+ wrapper->resize.handle = ew;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ suil_add_feature(features, &n_features, LV2_UI__parent, ew->cocoaView());
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/gtk2_in_qt4.cpp b/src/gtk2_in_qt4.cpp
new file mode 100644
index 0000000..546f7a9
--- /dev/null
+++ b/src/gtk2_in_qt4.cpp
@@ -0,0 +1,147 @@
+/*
+ Copyright 2011-2015 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <QX11EmbedContainer>
+#undef signals
+
+#include <gtk/gtk.h>
+#include <gdk/gdkx.h>
+
+#include "./suil_config.h"
+#include "./suil_internal.h"
+
+extern "C" {
+
+typedef struct _SuilGtk2InQt4Wrapper SuilGtk2InQt4Wrapper;
+
+struct _SuilGtk2InQt4Wrapper {
+ QX11EmbedContainer* host_widget;
+ QWidget* parent;
+ GtkWidget* plug;
+};
+
+static void
+on_size_request(GtkWidget* widget,
+ GtkRequisition* requisition,
+ gpointer user_data)
+{
+ QX11EmbedContainer* const wrap = (QX11EmbedContainer*)user_data;
+ wrap->setMinimumSize(requisition->width, requisition->height);
+}
+
+static void
+on_size_allocate(GtkWidget* widget,
+ GdkRectangle* allocation,
+ gpointer user_data)
+{
+ QX11EmbedContainer* const wrap = (QX11EmbedContainer*)user_data;
+ wrap->resize(allocation->width, allocation->height);
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ SuilGtk2InQt4Wrapper* impl = (SuilGtk2InQt4Wrapper*)wrapper->impl;
+
+ if (impl->plug) {
+ gtk_widget_destroy(impl->plug);
+ }
+
+ if (impl->host_widget) {
+ delete impl->host_widget;
+ }
+
+ free(impl);
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilGtk2InQt4Wrapper* const impl = (SuilGtk2InQt4Wrapper*)wrapper->impl;
+ QWidget* root = static_cast<QWidget*>(impl->parent);
+ QX11EmbedContainer* const wrap = new QX11EmbedContainer(root);
+ GtkWidget* const plug = gtk_plug_new(wrap->winId());
+ GtkWidget* const widget = (GtkWidget*)instance->ui_widget;
+
+ gtk_container_add(GTK_CONTAINER(plug), widget);
+ gtk_widget_show_all(plug);
+
+#ifdef SUIL_OLD_GTK
+ wrap->resize(widget->allocation.width, widget->allocation.height);
+#else
+ GtkAllocation alloc;
+ gtk_widget_get_allocation(widget, &alloc);
+ wrap->resize(alloc.width, alloc.height);
+#endif
+
+ g_signal_connect(
+ G_OBJECT(plug), "size-request", G_CALLBACK(on_size_request), wrap);
+
+ g_signal_connect(
+ G_OBJECT(plug), "size-allocate", G_CALLBACK(on_size_allocate), wrap);
+
+ impl->host_widget = wrap;
+ impl->plug = plug;
+ instance->host_widget = wrap;
+
+ return 0;
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ /* We have to open libgtk here, so Gtk type symbols are present and will be
+ found by the introspection stuff. This is required at least to make
+ GtkBuilder use in UIs work, otherwise they will cause "Invalid object
+ type" errors.
+ */
+ if (!host->gtk_lib) {
+ dlerror();
+ host->gtk_lib = dlopen(SUIL_GTK2_LIB_NAME, RTLD_LAZY|RTLD_GLOBAL);
+ if (!host->gtk_lib) {
+ SUIL_ERRORF("Failed to open %s (%s)\n",
+ SUIL_GTK2_LIB_NAME, dlerror());
+ return NULL;
+ }
+ gtk_init(NULL, NULL);
+ }
+
+ /* Create wrapper implementation. */
+ SuilGtk2InQt4Wrapper* const impl = (SuilGtk2InQt4Wrapper*)
+ calloc(1, sizeof(SuilGtk2InQt4Wrapper));
+
+ /* Set parent widget if given. */
+ for (unsigned i = 0; i < n_features; ++i) {
+ if (!strcmp((*features)[i]->URI, LV2_UI__parent)) {
+ impl->parent = static_cast<QWidget*>((*features)[i]->data);
+ }
+ }
+
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+ wrapper->impl = impl;
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/gtk2_in_qt5.cpp b/src/gtk2_in_qt5.cpp
new file mode 100644
index 0000000..77af460
--- /dev/null
+++ b/src/gtk2_in_qt5.cpp
@@ -0,0 +1,157 @@
+/*
+ Copyright 2011-2015 David Robillard <http://drobilla.net>
+ Copyright 2015 Rui Nuno Capela <rncbc@rncbc.org>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <QVBoxLayout>
+#include <QWidget>
+#include <QWindow>
+
+#undef signals
+
+#include <gtk/gtk.h>
+
+#include "./suil_config.h"
+#include "./suil_internal.h"
+
+extern "C" {
+
+typedef struct _SuilGtk2InQt5Wrapper SuilGtk2InQt5Wrapper;
+
+struct _SuilGtk2InQt5Wrapper {
+ QWidget* host_widget;
+ QWindow* window;
+ GtkWidget* plug;
+};
+
+static void
+on_size_request(GtkWidget* widget,
+ GtkRequisition* requisition,
+ gpointer user_data)
+{
+ QWidget* const wrap = (QWidget*)user_data;
+ wrap->setMinimumSize(requisition->width, requisition->height);
+}
+
+static void
+on_size_allocate(GtkWidget* widget,
+ GdkRectangle* allocation,
+ gpointer user_data)
+{
+ QWidget* const wrap = (QWidget*)user_data;
+ wrap->resize(allocation->width, allocation->height);
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ SuilGtk2InQt5Wrapper* impl = (SuilGtk2InQt5Wrapper*)wrapper->impl;
+
+ if (impl->window) {
+ impl->window->setParent(NULL);
+ delete impl->window;
+ }
+
+ if (impl->plug) {
+ gtk_widget_destroy(impl->plug);
+ }
+
+ if (impl->host_widget) {
+ delete impl->host_widget;
+ }
+
+ free(impl);
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilGtk2InQt5Wrapper* const impl = (SuilGtk2InQt5Wrapper*)wrapper->impl;
+ QWidget* const wrap = new QWidget(NULL, Qt::Window);
+ GtkWidget* const plug = gtk_plug_new(0);
+ GtkWidget* const widget = (GtkWidget*)instance->ui_widget;
+
+ gtk_container_add(GTK_CONTAINER(plug), widget);
+ gtk_widget_show_all(plug);
+
+ const WId wid = (const WId)gtk_plug_get_id((GtkPlug*)plug);
+ QWindow* window = QWindow::fromWinId(wid);
+ QWidget* container = QWidget::createWindowContainer(window, wrap);
+ QVBoxLayout* layout = new QVBoxLayout();
+ layout->setMargin(0);
+ layout->setSpacing(0);
+ layout->addWidget(container);
+ wrap->setLayout(layout);
+
+#ifdef SUIL_OLD_GTK
+ wrap->resize(widget->allocation.width, widget->allocation.height);
+#else
+ GtkAllocation alloc;
+ gtk_widget_get_allocation(widget, &alloc);
+ wrap->resize(alloc.width, alloc.height);
+#endif
+
+ g_signal_connect(
+ G_OBJECT(plug), "size-request", G_CALLBACK(on_size_request), wrap);
+
+ g_signal_connect(
+ G_OBJECT(plug), "size-allocate", G_CALLBACK(on_size_allocate), wrap);
+
+ impl->host_widget = wrap;
+ impl->window = window;
+ impl->plug = plug;
+ instance->host_widget = wrap;
+
+ return 0;
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ /* We have to open libgtk here, so Gtk type symbols are present and will be
+ found by the introspection stuff. This is required at least to make
+ GtkBuilder use in UIs work, otherwise they will cause "Invalid object
+ type" errors.
+ */
+ if (!host->gtk_lib) {
+ dlerror();
+ host->gtk_lib = dlopen(SUIL_GTK2_LIB_NAME, RTLD_LAZY|RTLD_GLOBAL);
+ if (!host->gtk_lib) {
+ SUIL_ERRORF("Failed to open %s (%s)\n",
+ SUIL_GTK2_LIB_NAME, dlerror());
+ return NULL;
+ }
+ gtk_init(NULL, NULL);
+ }
+
+ /* Create wrapper implementation. */
+ SuilGtk2InQt5Wrapper* const impl = (SuilGtk2InQt5Wrapper*)
+ calloc(1, sizeof(SuilGtk2InQt5Wrapper));
+
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+ wrapper->impl = impl;
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/host.c b/src/host.c
new file mode 100644
index 0000000..f64f9c5
--- /dev/null
+++ b/src/host.c
@@ -0,0 +1,90 @@
+/*
+ Copyright 2011-2017 David Robillard <http://drobilla.net>
+ Copyright 2017 Stefan Westerfeld <stefan@space.twc.de>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include "./suil_internal.h"
+
+int suil_argc = 0;
+char** suil_argv = NULL;
+
+SUIL_API
+SuilHost*
+suil_host_new(SuilPortWriteFunc write_func,
+ SuilPortIndexFunc index_func,
+ SuilPortSubscribeFunc subscribe_func,
+ SuilPortUnsubscribeFunc unsubscribe_func)
+{
+ SuilHost* host = (SuilHost*)calloc(1, sizeof(struct SuilHostImpl));
+ host->write_func = write_func;
+ host->index_func = index_func;
+ host->subscribe_func = subscribe_func;
+ host->unsubscribe_func = unsubscribe_func;
+ host->argc = suil_argc;
+ host->argv = suil_argv;
+ return host;
+}
+
+SUIL_API
+void
+suil_host_set_touch_func(SuilHost* host,
+ SuilTouchFunc touch_func)
+{
+ host->touch_func = touch_func;
+}
+
+SUIL_API
+void
+suil_host_free(SuilHost* host)
+{
+ if (host) {
+ if (host->gtk_lib) {
+ dlclose(host->gtk_lib);
+ }
+ free(host);
+ }
+}
+
+#ifdef SUIL_WITH_X11
+static void
+suil_load_init_module(const char* module_name)
+{
+ void* const lib = suil_open_module(module_name);
+ if (!lib) {
+ return;
+ }
+
+ SuilVoidFunc init_func = suil_dlfunc(lib, "suil_host_init");
+ if (init_func) {
+ (*init_func)();
+ } else {
+ SUIL_ERRORF("Corrupt init module %s\n", module_name);
+ }
+
+ dlclose(lib);
+}
+#endif
+
+SUIL_API
+void
+suil_init(int* argc, char*** argv, SuilArg key, ...)
+{
+ suil_argc = argc ? *argc : 0;
+ suil_argv = argv ? *argv : NULL;
+
+#ifdef SUIL_WITH_X11
+ suil_load_init_module("suil_x11");
+#endif
+}
diff --git a/src/instance.c b/src/instance.c
new file mode 100644
index 0000000..d365ca2
--- /dev/null
+++ b/src/instance.c
@@ -0,0 +1,390 @@
+/*
+ Copyright 2007-2017 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <assert.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "./suil_config.h"
+#include "./suil_internal.h"
+
+#define GTK2_UI_URI LV2_UI__GtkUI
+#define GTK3_UI_URI LV2_UI__Gtk3UI
+#define QT4_UI_URI LV2_UI__Qt4UI
+#define QT5_UI_URI LV2_UI__Qt5UI
+#define X11_UI_URI LV2_UI__X11UI
+#define WIN_UI_URI LV2_UI_PREFIX "WindowsUI"
+#define COCOA_UI_URI LV2_UI__CocoaUI
+
+SUIL_API
+unsigned
+suil_ui_supported(const char* host_type_uri,
+ const char* ui_type_uri)
+{
+ enum {
+ SUIL_WRAPPING_UNSUPPORTED = 0,
+ SUIL_WRAPPING_NATIVE = 1,
+ SUIL_WRAPPING_EMBEDDED = 2
+ };
+ if (!strcmp(host_type_uri, ui_type_uri)) {
+ return SUIL_WRAPPING_NATIVE;
+ } else if ((!strcmp(host_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, QT4_UI_URI))
+ || (!strcmp(host_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, QT5_UI_URI))
+ || (!strcmp(host_type_uri, QT4_UI_URI)
+ && !strcmp(ui_type_uri, GTK2_UI_URI))
+ || (!strcmp(host_type_uri, QT5_UI_URI)
+ && !strcmp(ui_type_uri, GTK2_UI_URI))
+ || (!strcmp(host_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI))
+ || (!strcmp(host_type_uri, GTK3_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI))
+ || (!strcmp(host_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, WIN_UI_URI))
+ || (!strcmp(host_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, COCOA_UI_URI))
+ || (!strcmp(host_type_uri, QT4_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI))
+ || (!strcmp(host_type_uri, QT5_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI))
+ || (!strcmp(host_type_uri, QT5_UI_URI)
+ && !strcmp(ui_type_uri, COCOA_UI_URI))) {
+ return SUIL_WRAPPING_EMBEDDED;
+ } else {
+ return SUIL_WRAPPING_UNSUPPORTED;
+ }
+}
+
+static SuilWrapper*
+open_wrapper(SuilHost* host,
+ const char* container_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ const char* module_name = NULL;
+#ifdef SUIL_WITH_GTK2_IN_QT4
+ if (!strcmp(container_type_uri, QT4_UI_URI)
+ && !strcmp(ui_type_uri, GTK2_UI_URI)) {
+ module_name = "suil_gtk2_in_qt4";
+ }
+#endif
+#ifdef SUIL_WITH_GTK2_IN_QT5
+ if (!strcmp(container_type_uri, QT5_UI_URI)
+ && !strcmp(ui_type_uri, GTK2_UI_URI)) {
+ module_name = "suil_gtk2_in_qt5";
+ }
+#endif
+#ifdef SUIL_WITH_QT4_IN_GTK2
+ if (!strcmp(container_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, QT4_UI_URI)) {
+ module_name = "suil_qt4_in_gtk2";
+ }
+#endif
+#ifdef SUIL_WITH_QT5_IN_GTK2
+ if (!strcmp(container_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, QT5_UI_URI)) {
+ module_name = "suil_qt5_in_gtk2";
+ }
+#endif
+#ifdef SUIL_WITH_X11_IN_GTK2
+ if (!strcmp(container_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI)) {
+ module_name = "suil_x11_in_gtk2";
+ }
+#endif
+#ifdef SUIL_WITH_X11_IN_GTK3
+ if (!strcmp(container_type_uri, GTK3_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI)) {
+ module_name = "suil_x11_in_gtk3";
+ }
+#endif
+#ifdef SUIL_WITH_WIN_IN_GTK2
+ if (!strcmp(container_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, WIN_UI_URI)) {
+ module_name = "suil_win_in_gtk2";
+ }
+#endif
+#ifdef SUIL_WITH_COCOA_IN_GTK2
+ if (!strcmp(container_type_uri, GTK2_UI_URI)
+ && !strcmp(ui_type_uri, COCOA_UI_URI)) {
+ module_name = "suil_cocoa_in_gtk2";
+ }
+#endif
+#ifdef SUIL_WITH_X11_IN_QT4
+ if (!strcmp(container_type_uri, QT4_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI)) {
+ module_name = "suil_x11_in_qt4";
+ }
+#endif
+#ifdef SUIL_WITH_X11_IN_QT5
+ if (!strcmp(container_type_uri, QT5_UI_URI)
+ && !strcmp(ui_type_uri, X11_UI_URI)) {
+ module_name = "suil_x11_in_qt5";
+ }
+#endif
+#ifdef SUIL_WITH_COCOA_IN_QT5
+ if (!strcmp(container_type_uri, QT5_UI_URI)
+ && !strcmp(ui_type_uri, COCOA_UI_URI)) {
+ module_name = "suil_cocoa_in_qt5";
+ }
+#endif
+
+ if (!module_name) {
+ SUIL_ERRORF("Unable to wrap UI type <%s> as type <%s>\n",
+ ui_type_uri, container_type_uri);
+ return NULL;
+ }
+
+ void* const lib = suil_open_module(module_name);
+ if (!lib) {
+ return NULL;
+ }
+
+ SuilWrapperNewFunc wrapper_new = (SuilWrapperNewFunc)suil_dlfunc(
+ lib, "suil_wrapper_new");
+
+ SuilWrapper* wrapper = wrapper_new
+ ? wrapper_new(host,
+ container_type_uri,
+ ui_type_uri,
+ features,
+ n_features)
+ : NULL;
+
+ if (wrapper) {
+ wrapper->lib = lib;
+ } else {
+ SUIL_ERRORF("Corrupt wrap module %s\n", module_name);
+ dlclose(lib);
+ }
+
+ return wrapper;
+}
+
+SUIL_API
+SuilInstance*
+suil_instance_new(SuilHost* host,
+ SuilController controller,
+ const char* container_type_uri,
+ const char* plugin_uri,
+ const char* ui_uri,
+ const char* ui_type_uri,
+ const char* ui_bundle_path,
+ const char* ui_binary_path,
+ const LV2_Feature* const* features)
+{
+ // Open UI library
+ dlerror();
+ void* lib = dlopen(ui_binary_path, RTLD_NOW);
+ if (!lib) {
+ SUIL_ERRORF("Unable to open UI library %s (%s)\n",
+ ui_binary_path, dlerror());
+ return NULL;
+ }
+
+ // Get discovery function
+ LV2UI_DescriptorFunction df = (LV2UI_DescriptorFunction)
+ suil_dlfunc(lib, "lv2ui_descriptor");
+ if (!df) {
+ SUIL_ERRORF("Broken LV2 UI %s (no lv2ui_descriptor symbol found)\n",
+ ui_binary_path);
+ dlclose(lib);
+ return NULL;
+ }
+
+ // Get UI descriptor
+ const LV2UI_Descriptor* descriptor = NULL;
+ for (uint32_t i = 0; true; ++i) {
+ const LV2UI_Descriptor* ld = df(i);
+ if (!ld) {
+ break;
+ } else if (!strcmp(ld->URI, ui_uri)) {
+ descriptor = ld;
+ break;
+ }
+ }
+ if (!descriptor) {
+ SUIL_ERRORF("Failed to find descriptor for <%s> in %s\n",
+ ui_uri, ui_binary_path);
+ dlclose(lib);
+ return NULL;
+ }
+
+ // Create SuilInstance
+ SuilInstance* instance = (SuilInstance*)calloc(1, sizeof(SuilInstance));
+ if (!instance) {
+ SUIL_ERRORF("Failed to allocate memory for <%s> instance\n", ui_uri);
+ dlclose(lib);
+ return NULL;
+ }
+
+ instance->lib_handle = lib;
+ instance->descriptor = descriptor;
+
+ // Make UI features array
+ instance->features = (LV2_Feature**)malloc(sizeof(LV2_Feature*));
+ instance->features[0] = NULL;
+
+ // Copy user provided features
+ const LV2_Feature* const* fi = features;
+ unsigned n_features = 0;
+ while (fi && *fi) {
+ const LV2_Feature* f = *fi++;
+ suil_add_feature(&instance->features, &n_features, f->URI, f->data);
+ }
+
+ // Add additional features implemented by SuilHost functions
+ if (host->index_func) {
+ instance->port_map.handle = controller;
+ instance->port_map.port_index = host->index_func;
+ suil_add_feature(&instance->features, &n_features,
+ LV2_UI__portMap, &instance->port_map);
+ }
+ if (host->subscribe_func && host->unsubscribe_func) {
+ instance->port_subscribe.handle = controller;
+ instance->port_subscribe.subscribe = host->subscribe_func;
+ instance->port_subscribe.unsubscribe = host->unsubscribe_func;
+ suil_add_feature(&instance->features, &n_features,
+ LV2_UI__portSubscribe, &instance->port_subscribe);
+ }
+ if (host->touch_func) {
+ instance->touch.handle = controller;
+ instance->touch.touch = host->touch_func;
+ suil_add_feature(&instance->features, &n_features,
+ LV2_UI__touch, &instance->touch);
+ }
+
+ // Open wrapper (this may add additional features)
+ if (container_type_uri && strcmp(container_type_uri, ui_type_uri)) {
+ instance->wrapper = open_wrapper(host,
+ container_type_uri, ui_type_uri,
+ &instance->features, n_features);
+ if (!instance->wrapper) {
+ suil_instance_free(instance);
+ return NULL;
+ }
+ }
+
+ // Instantiate UI
+ instance->handle = descriptor->instantiate(
+ descriptor,
+ plugin_uri,
+ ui_bundle_path,
+ host->write_func,
+ controller,
+ &instance->ui_widget,
+ (const LV2_Feature* const*)instance->features);
+
+ // Failed to instantiate UI
+ if (!instance->handle) {
+ SUIL_ERRORF("Failed to instantiate UI <%s> in %s\n",
+ ui_uri, ui_binary_path);
+ suil_instance_free(instance);
+ return NULL;
+ }
+
+ if (instance->wrapper) {
+ if (instance->wrapper->wrap(instance->wrapper, instance)) {
+ SUIL_ERRORF("Failed to wrap UI <%s> in type <%s>\n",
+ ui_uri, container_type_uri);
+ suil_instance_free(instance);
+ return NULL;
+ }
+ } else {
+ instance->host_widget = instance->ui_widget;
+ }
+
+ return instance;
+}
+
+SUIL_API
+void
+suil_instance_free(SuilInstance* instance)
+{
+ if (instance) {
+ for (unsigned i = 0; instance->features[i]; ++i) {
+ free(instance->features[i]);
+ }
+ free(instance->features);
+
+ // Call wrapper free function to destroy widgets and drop references
+ if (instance->wrapper && instance->wrapper->free) {
+ instance->wrapper->free(instance->wrapper);
+ }
+
+ // Call cleanup to destroy UI (if it still exists at this point)
+ if (instance->handle) {
+ instance->descriptor->cleanup(instance->handle);
+ }
+
+ dlclose(instance->lib_handle);
+
+ // Close libraries and free everything
+ if (instance->wrapper) {
+#ifndef _WIN32
+ // Never unload modules on windows, causes mysterious segfaults
+ dlclose(instance->wrapper->lib);
+#endif
+ free(instance->wrapper);
+ }
+ free(instance);
+ }
+}
+
+SUIL_API
+SuilHandle
+suil_instance_get_handle(SuilInstance* instance)
+{
+ return instance->handle;
+}
+
+SUIL_API
+LV2UI_Widget
+suil_instance_get_widget(SuilInstance* instance)
+{
+ return instance->host_widget;
+}
+
+SUIL_API
+void
+suil_instance_port_event(SuilInstance* instance,
+ uint32_t port_index,
+ uint32_t buffer_size,
+ uint32_t format,
+ const void* buffer)
+{
+ if (instance->descriptor->port_event) {
+ instance->descriptor->port_event(instance->handle,
+ port_index,
+ buffer_size,
+ format,
+ buffer);
+ }
+}
+
+SUIL_API
+const void*
+suil_instance_extension_data(SuilInstance* instance,
+ const char* uri)
+{
+ if (instance->descriptor->extension_data) {
+ return instance->descriptor->extension_data(uri);
+ }
+ return NULL;
+}
diff --git a/src/qt4_in_gtk2.cpp b/src/qt4_in_gtk2.cpp
new file mode 100644
index 0000000..fbec381
--- /dev/null
+++ b/src/qt4_in_gtk2.cpp
@@ -0,0 +1,154 @@
+/*
+ Copyright 2011-2017 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <gtk/gtk.h>
+
+#include <QApplication>
+#include <QVBoxLayout>
+#include <QX11EmbedWidget>
+
+#include "./suil_internal.h"
+
+extern "C" {
+
+#define SUIL_TYPE_QT_WRAPPER (suil_qt_wrapper_get_type())
+#define SUIL_QT_WRAPPER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SUIL_TYPE_QT_WRAPPER, SuilQtWrapper))
+
+typedef struct _SuilQtWrapper SuilQtWrapper;
+typedef struct _SuilQtWrapperClass SuilQtWrapperClass;
+
+struct _SuilQtWrapper {
+ GtkSocket socket;
+ QApplication* app;
+ QX11EmbedWidget* qembed;
+ SuilWrapper* wrapper;
+ SuilInstance* instance;
+};
+
+struct _SuilQtWrapperClass {
+ GtkSocketClass parent_class;
+};
+
+GType suil_qt_wrapper_get_type(void); // Accessor for SUIL_TYPE_QT_WRAPPER
+
+G_DEFINE_TYPE(SuilQtWrapper, suil_qt_wrapper, GTK_TYPE_SOCKET)
+
+static void
+suil_qt_wrapper_finalize(GObject* gobject)
+{
+ SuilQtWrapper* const self = SUIL_QT_WRAPPER(gobject);
+
+ if (self->instance->handle) {
+ self->instance->descriptor->cleanup(self->instance->handle);
+ self->instance->handle = NULL;
+ }
+
+ delete self->qembed;
+
+ self->qembed = NULL;
+ self->app = NULL;
+ self->wrapper->impl = NULL;
+
+ G_OBJECT_CLASS(suil_qt_wrapper_parent_class)->finalize(gobject);
+}
+
+static void
+suil_qt_wrapper_class_init(SuilQtWrapperClass* klass)
+{
+ GObjectClass* const gobject_class = G_OBJECT_CLASS(klass);
+
+ gobject_class->finalize = suil_qt_wrapper_finalize;
+}
+
+static void
+suil_qt_wrapper_init(SuilQtWrapper* self)
+{
+ self->app = NULL;
+ self->qembed = NULL;
+ self->instance = NULL;
+}
+
+static void
+suil_qt_wrapper_realize(GtkWidget* w, gpointer data)
+{
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(w);
+ GtkSocket* const s = GTK_SOCKET(w);
+
+ gtk_socket_add_id(s, wrap->qembed->winId());
+ wrap->qembed->show();
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(wrapper->impl);
+
+ wrap->qembed = new QX11EmbedWidget();
+ wrap->wrapper = wrapper;
+ wrap->instance = instance;
+
+ QWidget* qwidget = (QWidget*)instance->ui_widget;
+ QVBoxLayout* layout = new QVBoxLayout(wrap->qembed);
+ layout->addWidget(qwidget);
+
+ qwidget->setParent(wrap->qembed);
+
+ g_signal_connect_after(G_OBJECT(wrap), "realize",
+ G_CALLBACK(suil_qt_wrapper_realize), NULL);
+
+ instance->host_widget = GTK_WIDGET(wrap);
+
+ return 0;
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ if (wrapper->impl) {
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(wrapper->impl);
+ gtk_object_destroy(GTK_OBJECT(wrap));
+ }
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(
+ g_object_new(SUIL_TYPE_QT_WRAPPER, NULL));
+
+ if (qApp) {
+ wrap->app = qApp;
+ } else {
+ wrap->app = new QApplication(host->argc, host->argv, true);
+ }
+
+ wrap->wrapper = NULL;
+ wrapper->impl = wrap;
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/qt5_in_gtk2.cpp b/src/qt5_in_gtk2.cpp
new file mode 100644
index 0000000..a03e8a7
--- /dev/null
+++ b/src/qt5_in_gtk2.cpp
@@ -0,0 +1,237 @@
+/*
+ Copyright 2011-2017 David Robillard <http://drobilla.net>
+ Copyright 2018 Rui Nuno Capela <rncbc@rncbc.org>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <gtk/gtk.h>
+
+#include <QApplication>
+#include <QVBoxLayout>
+#include <QWidget>
+#include <QWindow>
+
+#include "lv2/lv2plug.in/ns/ext/options/options.h"
+#include "lv2/lv2plug.in/ns/ext/urid/urid.h"
+
+#include "./suil_internal.h"
+
+extern "C" {
+
+#define SUIL_TYPE_QT_WRAPPER (suil_qt_wrapper_get_type())
+#define SUIL_QT_WRAPPER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SUIL_TYPE_QT_WRAPPER, SuilQtWrapper))
+
+typedef struct _SuilQtWrapper SuilQtWrapper;
+typedef struct _SuilQtWrapperClass SuilQtWrapperClass;
+
+struct _SuilQtWrapper {
+ GtkSocket socket;
+ QApplication* app;
+ QWidget* qembed;
+ SuilWrapper* wrapper;
+ SuilInstance* instance;
+ const LV2UI_Idle_Interface* idle_iface;
+ guint idle_id;
+ guint idle_ms;
+};
+
+struct _SuilQtWrapperClass {
+ GtkSocketClass parent_class;
+};
+
+GType suil_qt_wrapper_get_type(void); // Accessor for SUIL_TYPE_QT_WRAPPER
+
+G_DEFINE_TYPE(SuilQtWrapper, suil_qt_wrapper, GTK_TYPE_SOCKET)
+
+static void
+suil_qt_wrapper_finalize(GObject* gobject)
+{
+ SuilQtWrapper* const self = SUIL_QT_WRAPPER(gobject);
+
+ if (self->idle_id) {
+ g_source_remove(self->idle_id);
+ self->idle_id = 0;
+ }
+
+ if (self->instance->handle) {
+ self->instance->descriptor->cleanup(self->instance->handle);
+ self->instance->handle = NULL;
+ }
+
+ delete self->qembed;
+
+ self->qembed = NULL;
+ self->app = NULL;
+ self->idle_iface = NULL;
+ self->wrapper->impl = NULL;
+
+ G_OBJECT_CLASS(suil_qt_wrapper_parent_class)->finalize(gobject);
+}
+
+static void
+suil_qt_wrapper_class_init(SuilQtWrapperClass* klass)
+{
+ GObjectClass* const gobject_class = G_OBJECT_CLASS(klass);
+
+ gobject_class->finalize = suil_qt_wrapper_finalize;
+}
+
+static void
+suil_qt_wrapper_init(SuilQtWrapper* self)
+{
+ self->app = NULL;
+ self->qembed = NULL;
+ self->wrapper = NULL;
+ self->instance = NULL;
+ self->idle_iface = NULL;
+ self->idle_id = 0;
+ self->idle_ms = 1000 / 30; // 30 Hz default
+}
+
+static void
+suil_qt_wrapper_realize(GtkWidget* w, gpointer data)
+{
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(w);
+ GtkSocket* const s = GTK_SOCKET(w);
+ const WId id = (const WId)gtk_socket_get_id(s);
+
+ wrap->qembed->winId();
+ wrap->qembed->windowHandle()->setParent(QWindow::fromWinId(id));
+ wrap->qembed->show();
+}
+
+static int
+suil_qt_wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ gtk_widget_set_size_request(GTK_WIDGET(handle), width, height);
+
+ return 0;
+}
+
+static gboolean
+suil_qt_wrapper_idle(void* data)
+{
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(data);
+
+ if (wrap->idle_iface) {
+ wrap->idle_iface->idle(wrap->instance->handle);
+ return TRUE; // Continue calling
+ }
+
+ return FALSE;
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(wrapper->impl);
+
+ wrap->qembed = new QWidget();
+ wrap->wrapper = wrapper;
+ wrap->instance = instance;
+
+ QWidget* qwidget = (QWidget*)instance->ui_widget;
+ QVBoxLayout* layout = new QVBoxLayout();
+ layout->setMargin(0);
+ layout->setSpacing(0);
+ layout->addWidget(qwidget);
+
+ wrap->qembed->setLayout(layout);
+
+ g_signal_connect_after(G_OBJECT(wrap), "realize",
+ G_CALLBACK(suil_qt_wrapper_realize), NULL);
+
+ instance->host_widget = GTK_WIDGET(wrap);
+
+ const LV2UI_Idle_Interface* idle_iface = NULL;
+ if (instance->descriptor->extension_data) {
+ idle_iface = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ }
+
+ if (idle_iface) {
+ wrap->idle_iface = idle_iface;
+ wrap->idle_id = g_timeout_add(
+ wrap->idle_ms, suil_qt_wrapper_idle, wrap);
+ }
+
+ return 0;
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ if (wrapper->impl) {
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(wrapper->impl);
+ gtk_object_destroy(GTK_OBJECT(wrap));
+ }
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ SuilQtWrapper* const wrap = SUIL_QT_WRAPPER(
+ g_object_new(SUIL_TYPE_QT_WRAPPER, NULL));
+
+ if (qApp) {
+ wrap->app = qApp;
+ } else {
+ wrap->app = new QApplication(host->argc, host->argv, true);
+ }
+
+ wrap->wrapper = NULL;
+ wrapper->impl = wrap;
+
+ wrapper->resize.handle = wrap;
+ wrapper->resize.ui_resize = suil_qt_wrapper_resize;
+
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ // Scan for URID map and options
+ LV2_URID_Map* map = NULL;
+ LV2_Options_Option* options = NULL;
+ for (LV2_Feature** f = *features; *f && (!map || !options); ++f) {
+ if (!strcmp((*f)->URI, LV2_OPTIONS__options)) {
+ options = (LV2_Options_Option*)(*f)->data;
+ } else if (!strcmp((*f)->URI, LV2_URID__map)) {
+ map = (LV2_URID_Map*)(*f)->data;
+ }
+ }
+
+ if (map && options) {
+ // Set UI update rate if given
+ LV2_URID ui_updateRate = map->map(map->handle, LV2_UI__updateRate);
+ for (LV2_Options_Option* o = options; o->key; ++o) {
+ if (o->key == ui_updateRate) {
+ wrap->idle_ms = 1000.0f / *(const float*)o->value;
+ break;
+ }
+ }
+ }
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/suil_internal.h b/src/suil_internal.h
new file mode 100644
index 0000000..ecafbcf
--- /dev/null
+++ b/src/suil_internal.h
@@ -0,0 +1,185 @@
+/*
+ Copyright 2007-2017 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#ifndef SUIL_INTERNAL_H
+#define SUIL_INTERNAL_H
+
+#include <assert.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#ifdef _WIN32
+#include <windows.h>
+#define dlopen(path, flags) LoadLibrary(path)
+#define dlclose(lib) FreeLibrary((HMODULE)lib)
+#define inline __inline
+#define snprintf _snprintf
+static inline char* dlerror(void) { return "Unknown error"; }
+#else
+#include <dlfcn.h>
+#endif
+
+#include "lv2/lv2plug.in/ns/extensions/ui/ui.h"
+
+#include "suil/suil.h"
+#include "./suil_config.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#define SUIL_ERRORF(fmt, ...) fprintf(stderr, "suil error: " fmt, __VA_ARGS__)
+
+struct SuilHostImpl {
+ SuilPortWriteFunc write_func;
+ SuilPortIndexFunc index_func;
+ SuilPortSubscribeFunc subscribe_func;
+ SuilPortUnsubscribeFunc unsubscribe_func;
+ SuilTouchFunc touch_func;
+ void* gtk_lib;
+ int argc;
+ char** argv;
+};
+
+struct _SuilWrapper;
+
+typedef void (*SuilWrapperFreeFunc)(struct _SuilWrapper*);
+
+typedef int (*SuilWrapperWrapFunc)(struct _SuilWrapper* wrapper,
+ SuilInstance* instance);
+
+typedef struct _SuilWrapper {
+ SuilWrapperWrapFunc wrap;
+ SuilWrapperFreeFunc free;
+ void* lib;
+ void* impl;
+ LV2UI_Resize resize;
+} SuilWrapper;
+
+struct SuilInstanceImpl {
+ void* lib_handle;
+ const LV2UI_Descriptor* descriptor;
+ LV2UI_Handle handle;
+ SuilWrapper* wrapper;
+ LV2_Feature** features;
+ LV2UI_Port_Map port_map;
+ LV2UI_Port_Subscribe port_subscribe;
+ LV2UI_Touch touch;
+ SuilWidget ui_widget;
+ SuilWidget host_widget;
+};
+
+/**
+ The type of the suil_wrapper_new entry point in a wrapper module.
+
+ This constructs a SuilWrapper which contains everything necessary
+ to wrap a widget, including a possibly extended features array to
+ be used for instantiating the UI.
+*/
+typedef SuilWrapper* (*SuilWrapperNewFunc)(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features);
+
+/** Prototype for suil_wrapper_new in each wrapper module. */
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features);
+
+/** Prototype for suil_host_init in each init module. */
+SUIL_LIB_EXPORT
+void
+suil_host_init(void);
+
+/** Dynamically load the suil module with the given name. */
+static inline void*
+suil_open_module(const char* module_name)
+{
+ const char* const env_dir = getenv("SUIL_MODULE_DIR");
+ const char* const mod_dir = env_dir ? env_dir : SUIL_MODULE_DIR;
+ const size_t path_len = strlen(mod_dir)
+ + strlen(SUIL_DIR_SEP SUIL_MODULE_PREFIX SUIL_MODULE_EXT)
+ + strlen(module_name)
+ + 2;
+
+ char* const path = (char*)calloc(path_len, 1);
+ snprintf(path, path_len, "%s%s%s%s%s",
+ mod_dir, SUIL_DIR_SEP,
+ SUIL_MODULE_PREFIX, module_name, SUIL_MODULE_EXT);
+
+ dlerror();
+ void* lib = dlopen(path, RTLD_NOW);
+ if (!lib) {
+ SUIL_ERRORF("Failed to open module %s (%s)\n", path, dlerror());
+ }
+
+ free(path);
+ return lib;
+}
+
+typedef void (*SuilVoidFunc)(void);
+
+/** dlsym wrapper to return a function pointer (without annoying warning) */
+static inline SuilVoidFunc
+suil_dlfunc(void* handle, const char* symbol)
+{
+#ifdef _WIN32
+ return (SuilVoidFunc)GetProcAddress((HMODULE)handle, symbol);
+#else
+ typedef SuilVoidFunc (*VoidFuncGetter)(void*, const char*);
+ VoidFuncGetter dlfunc = (VoidFuncGetter)dlsym;
+ return dlfunc(handle, symbol);
+#endif
+}
+
+/** Add a feature to a (mutable) LV2 feature array. */
+static inline void
+suil_add_feature(LV2_Feature*** features,
+ unsigned* n,
+ const char* uri,
+ void* data)
+{
+ for (unsigned i = 0; i < *n && (*features)[i]; ++i) {
+ if (!strcmp((*features)[i]->URI, uri)) {
+ (*features)[i]->data = data;
+ return;
+ }
+ }
+
+ *features = (LV2_Feature**)realloc(*features,
+ sizeof(LV2_Feature*) * (*n + 2));
+
+ (*features)[*n] = (LV2_Feature*)malloc(sizeof(LV2_Feature));
+ (*features)[*n]->URI = uri;
+ (*features)[*n]->data = data;
+ (*features)[*n + 1] = NULL;
+ *n += 1;
+}
+
+extern int suil_argc;
+extern char** suil_argv;
+
+#ifdef __cplusplus
+} /* extern "C" */
+#endif
+
+#endif // SUIL_INTERNAL_H
diff --git a/src/win_in_gtk2.cpp b/src/win_in_gtk2.cpp
new file mode 100644
index 0000000..70c7c1a
--- /dev/null
+++ b/src/win_in_gtk2.cpp
@@ -0,0 +1,258 @@
+/*
+ Copyright 2011-2015 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <string.h>
+
+#include <gtk/gtk.h>
+#include <gdk/gdkwin32.h>
+
+#ifndef WM_MOUSEWHEEL
+# define WM_MOUSEWHEEL 0x020A
+#endif
+#ifndef WM_MOUSEHWHEEL
+# define WM_MOUSEHWHEEL 0x020E
+#endif
+
+#include "./suil_internal.h"
+
+#include "lv2/lv2plug.in/ns/ext/options/options.h"
+#include "lv2/lv2plug.in/ns/ext/urid/urid.h"
+
+extern "C" {
+
+#define SUIL_TYPE_WIN_WRAPPER (suil_win_wrapper_get_type())
+#define SUIL_WIN_WRAPPER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SUIL_TYPE_WIN_WRAPPER, SuilWinWrapper))
+
+typedef struct _SuilWinWrapper SuilWinWrapper;
+typedef struct _SuilWinWrapperClass SuilWinWrapperClass;
+
+struct _SuilWinWrapper {
+ GtkDrawingArea area;
+ SuilWrapper* wrapper;
+ SuilInstance* instance;
+ GdkWindow* flt_win;
+ const LV2UI_Idle_Interface* idle_iface;
+ guint idle_id;
+ guint idle_ms;
+};
+
+struct _SuilWinWrapperClass {
+ GtkDrawingAreaClass parent_class;
+};
+
+GType suil_win_wrapper_get_type(void); // Accessor for SUIL_TYPE_WIN_WRAPPER
+
+G_DEFINE_TYPE(SuilWinWrapper, suil_win_wrapper, GTK_TYPE_DRAWING_AREA)
+
+static void
+suil_win_wrapper_finalize(GObject* gobject)
+{
+ SuilWinWrapper* const self = SUIL_WIN_WRAPPER(gobject);
+
+ self->wrapper->impl = NULL;
+ self->instance = NULL;
+
+ G_OBJECT_CLASS(suil_win_wrapper_parent_class)->finalize(gobject);
+}
+
+static void
+suil_win_size_allocate(GtkWidget* widget, GtkAllocation* allocation)
+{
+ SuilWinWrapper* const self = SUIL_WIN_WRAPPER(widget);
+ g_return_if_fail(self != NULL);
+
+ widget->allocation = *allocation;
+ if (gtk_widget_get_realized(widget)) {
+ gdk_window_move_resize(widget->window,
+ allocation->x, allocation->y,
+ allocation->width, allocation->height);
+
+ RECT wr = { 0, 0, (long)allocation->width, (long)allocation->height };
+ AdjustWindowRectEx(&wr, WS_CHILD, FALSE, WS_EX_TOPMOST);
+
+ SetWindowPos((HWND)self->instance->ui_widget, HWND_NOTOPMOST,
+ 0, 0, wr.right - wr.left, wr.bottom - wr.top,
+ SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOOWNERZORDER|SWP_NOZORDER);
+ UpdateWindow((HWND)self->instance->ui_widget);
+ PostMessage((HWND)self->instance->ui_widget, WM_PAINT, 0, 0);
+ }
+}
+
+static void
+suil_win_wrapper_class_init(SuilWinWrapperClass* klass)
+{
+ GObjectClass* const gobject_class = G_OBJECT_CLASS(klass);
+ GtkWidgetClass* const widget_class = (GtkWidgetClass*)(klass);
+
+ widget_class->size_allocate = suil_win_size_allocate;
+ gobject_class->finalize = suil_win_wrapper_finalize;
+}
+
+static void
+suil_win_wrapper_init(SuilWinWrapper* self)
+{
+ self->instance = NULL;
+ self->flt_win = NULL;
+ self->idle_iface = NULL;
+ self->idle_ms = 1000 / 30; // 30 Hz default
+}
+
+static gboolean
+suil_win_wrapper_idle(void* data)
+{
+ SuilWinWrapper* const wrap = SUIL_WIN_WRAPPER(data);
+ wrap->idle_iface->idle(wrap->instance->handle);
+ return TRUE; // Continue calling
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ gtk_drawing_area_size(GTK_DRAWING_AREA(handle), width, height);
+ return 0;
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilWinWrapper* const wrap = SUIL_WIN_WRAPPER(wrapper->impl);
+
+ instance->host_widget = GTK_WIDGET(wrap);
+ wrap->wrapper = wrapper;
+ wrap->instance = instance;
+
+ const LV2UI_Idle_Interface* idle_iface = NULL;
+ if (instance->descriptor->extension_data) {
+ idle_iface = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ }
+ if (idle_iface) {
+ wrap->idle_iface = idle_iface;
+ wrap->idle_id = g_timeout_add (wrap->idle_ms, suil_win_wrapper_idle, wrap);
+ }
+
+ return 0;
+}
+
+static GdkFilterReturn
+event_filter(GdkXEvent* xevent, GdkEvent* event, gpointer data)
+{
+ SuilWinWrapper* wrap = (SuilWinWrapper*)data;
+ MSG* msg = (MSG*)xevent;
+ if (msg->message == WM_KEYDOWN || msg->message == WM_KEYUP) {
+ // Forward keyboard events to UI window
+ PostMessage((HWND)wrap->instance->ui_widget,
+ msg->message, msg->wParam, msg->lParam);
+ return GDK_FILTER_REMOVE;
+ } else if (msg->message == WM_MOUSEWHEEL || msg->message == WM_MOUSEHWHEEL) {
+ PostMessage((HWND)wrap->instance->ui_widget,
+ msg->message, msg->wParam, msg->lParam);
+ return GDK_FILTER_REMOVE;
+ }
+ return GDK_FILTER_CONTINUE;
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ if (wrapper->impl) {
+ SuilWinWrapper* const wrap = SUIL_WIN_WRAPPER(wrapper->impl);
+ if (wrap->idle_id) {
+ g_source_remove(wrap->idle_id);
+ wrap->idle_id = 0;
+ }
+
+ gdk_window_remove_filter(wrap->flt_win, event_filter, wrapper->impl);
+ gtk_object_destroy(GTK_OBJECT(wrap));
+ }
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ GtkWidget* parent = NULL;
+ for (unsigned i = 0; i < n_features; ++i) {
+ if (!strcmp((*features)[i]->URI, LV2_UI__parent)) {
+ parent = (GtkWidget*)(*features)[i]->data;
+ }
+ }
+
+ if (!GTK_CONTAINER(parent)) {
+ SUIL_ERRORF("No GtkContainer parent given for %s UI\n",
+ ui_type_uri);
+ return NULL;
+ }
+
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ SuilWinWrapper* const wrap = SUIL_WIN_WRAPPER(
+ g_object_new(SUIL_TYPE_WIN_WRAPPER, NULL));
+
+ wrap->wrapper = NULL;
+
+ wrapper->impl = wrap;
+ wrapper->resize.handle = wrap;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ gtk_container_add(GTK_CONTAINER(parent), GTK_WIDGET(wrap));
+ gtk_widget_set_can_focus(GTK_WIDGET(wrap), TRUE);
+ gtk_widget_set_sensitive(GTK_WIDGET(wrap), TRUE);
+ gtk_widget_realize(GTK_WIDGET(wrap));
+
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(wrap));
+
+ wrap->flt_win = gtk_widget_get_window(parent);
+ gdk_window_add_filter(wrap->flt_win, event_filter, wrap);
+
+ HWND parent_window = (HWND)GDK_WINDOW_HWND(window);
+ suil_add_feature(features, &n_features, LV2_UI__parent, parent_window);
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ // Scan for URID map and options
+ LV2_URID_Map* map = NULL;
+ LV2_Options_Option* options = NULL;
+ for (LV2_Feature** f = *features; *f && (!map || !options); ++f) {
+ if (!strcmp((*f)->URI, LV2_OPTIONS__options)) {
+ options = (LV2_Options_Option *)(*f)->data;
+ } else if (!strcmp((*f)->URI, LV2_URID__map)) {
+ map = (LV2_URID_Map *)(*f)->data;
+ }
+ }
+
+ if (map && options) {
+ // Set UI update rate if given
+ LV2_URID ui_updateRate = map->map(map->handle, LV2_UI__updateRate);
+ for (LV2_Options_Option* o = options; o->key; ++o) {
+ if (o->key == ui_updateRate) {
+ wrap->idle_ms = 1000.0f / *(const float*)o->value;
+ break;
+ }
+ }
+ }
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/x11.c b/src/x11.c
new file mode 100644
index 0000000..4c8a15a
--- /dev/null
+++ b/src/x11.c
@@ -0,0 +1,28 @@
+/*
+ Copyright 2017 David Robillard <http://drobilla.net>
+ Copyright 2017 Stefan Westerfeld <stefan@space.twc.de>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <X11/Xlib.h>
+
+#include "./suil_internal.h"
+
+SUIL_LIB_EXPORT
+void
+suil_host_init(void)
+{
+ // This must be called first for Qt5 in Gtk2 to function correctly
+ XInitThreads();
+}
diff --git a/src/x11_in_gtk2.c b/src/x11_in_gtk2.c
new file mode 100644
index 0000000..7ec9592
--- /dev/null
+++ b/src/x11_in_gtk2.c
@@ -0,0 +1,398 @@
+/*
+ Copyright 2011-2016 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <gdk/gdkx.h>
+#include <gtk/gtk.h>
+#include <string.h>
+
+#include "lv2/lv2plug.in/ns/ext/options/options.h"
+#include "lv2/lv2plug.in/ns/ext/urid/urid.h"
+
+#include "./suil_internal.h"
+
+#define SUIL_TYPE_X11_WRAPPER (suil_x11_wrapper_get_type())
+#define SUIL_X11_WRAPPER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SUIL_TYPE_X11_WRAPPER, SuilX11Wrapper))
+
+typedef struct _SuilX11Wrapper SuilX11Wrapper;
+typedef struct _SuilX11WrapperClass SuilX11WrapperClass;
+
+struct _SuilX11Wrapper {
+ GtkSocket socket;
+ GtkPlug* plug;
+ SuilWrapper* wrapper;
+ SuilInstance* instance;
+ const LV2UI_Idle_Interface* idle_iface;
+ guint idle_id;
+ guint idle_ms;
+};
+
+struct _SuilX11WrapperClass {
+ GtkSocketClass parent_class;
+};
+
+GType suil_x11_wrapper_get_type(void); // Accessor for SUIL_TYPE_X11_WRAPPER
+
+G_DEFINE_TYPE(SuilX11Wrapper, suil_x11_wrapper, GTK_TYPE_SOCKET)
+
+/**
+ Check if 'swallowed' subwindow is known to the X server.
+
+ Gdk/GTK can mark the window as realized, mapped and visible even though
+ there is no window-ID on the X server for it yet. Then,
+ suil_x11_on_size_allocate() will cause a "BadWinow" X error.
+*/
+static bool
+x_window_is_valid(SuilX11Wrapper* socket)
+{
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(socket->plug));
+ Window root = 0;
+ Window parent = 0;
+ Window* children = NULL;
+ unsigned childcount = 0;
+
+ XQueryTree(GDK_WINDOW_XDISPLAY(window),
+ GDK_WINDOW_XID(window),
+ &root, &parent, &children, &childcount);
+ for (unsigned i = 0; i < childcount; ++i) {
+ if (children[i] == (Window)socket->instance->ui_widget) {
+ XFree(children);
+ return true;
+ }
+ }
+ XFree(children);
+ return false;
+}
+
+static gboolean
+on_plug_removed(GtkSocket* sock, gpointer data)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(sock);
+
+ if (self->idle_id) {
+ g_source_remove(self->idle_id);
+ self->idle_id = 0;
+ }
+
+ if (self->instance->handle) {
+ self->instance->descriptor->cleanup(self->instance->handle);
+ self->instance->handle = NULL;
+ }
+
+ self->plug = NULL;
+ return TRUE;
+}
+
+static void
+suil_x11_wrapper_finalize(GObject* gobject)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(gobject);
+
+ self->wrapper->impl = NULL;
+
+ G_OBJECT_CLASS(suil_x11_wrapper_parent_class)->finalize(gobject);
+}
+
+static void
+suil_x11_wrapper_realize(GtkWidget* w)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(w);
+ GtkSocket* const socket = GTK_SOCKET(w);
+
+ if (GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->realize) {
+ GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->realize(w);
+ }
+
+ gtk_socket_add_id(socket, gtk_plug_get_id(wrap->plug));
+
+ gtk_widget_set_sensitive(GTK_WIDGET(wrap->plug), TRUE);
+ gtk_widget_set_can_focus(GTK_WIDGET(wrap->plug), TRUE);
+ gtk_widget_grab_focus(GTK_WIDGET(wrap->plug));
+}
+
+static void
+suil_x11_wrapper_show(GtkWidget* w)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(w);
+
+ if (GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->show) {
+ GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->show(w);
+ }
+
+ gtk_widget_show(GTK_WIDGET(wrap->plug));
+}
+
+static gboolean
+forward_key_event(SuilX11Wrapper* socket,
+ GdkEvent* gdk_event)
+{
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(socket->plug));
+ GdkScreen* screen = gdk_visual_get_screen(gdk_window_get_visual(window));
+
+ Window target_window;
+ if (gdk_event->any.window == window) {
+ // Event sent up to the plug window, forward it up to the parent
+ GtkWidget* widget = GTK_WIDGET(socket->instance->host_widget);
+ GdkWindow* parent = gtk_widget_get_parent_window(widget);
+ if (parent) {
+ target_window = GDK_WINDOW_XID(parent);
+ } else {
+ return FALSE; // Wrapper is a top-level window, do nothing
+ }
+ } else {
+ // Event sent anywhere else, send to the plugin
+ target_window = (Window)socket->instance->ui_widget;
+ }
+
+ XKeyEvent xev;
+ memset(&xev, 0, sizeof(xev));
+ xev.type = (gdk_event->type == GDK_KEY_PRESS) ? KeyPress : KeyRelease;
+ xev.root = GDK_WINDOW_XID(gdk_screen_get_root_window(screen));
+ xev.window = target_window;
+ xev.subwindow = None;
+ xev.time = gdk_event->key.time;
+ xev.state = gdk_event->key.state;
+ xev.keycode = gdk_event->key.hardware_keycode;
+
+ XSendEvent(GDK_WINDOW_XDISPLAY(window),
+ target_window,
+ False,
+ NoEventMask,
+ (XEvent*)&xev);
+
+ return (gdk_event->any.window != window);
+}
+
+static gboolean
+idle_size_request(gpointer user_data)
+{
+ GtkWidget* w = GTK_WIDGET(user_data);
+ gtk_widget_queue_resize(w);
+ return FALSE;
+}
+
+static void
+forward_size_request(SuilX11Wrapper* socket,
+ GtkAllocation* allocation)
+{
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(socket->plug));
+ if (x_window_is_valid(socket)) {
+ // Calculate allocation size constrained to X11 limits for widget
+ int width = allocation->width;
+ int height = allocation->height;
+ XSizeHints hints;
+ memset(&hints, 0, sizeof(hints));
+ XGetNormalHints(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ &hints);
+ if (hints.flags & PMaxSize) {
+ width = MIN(width, hints.max_width);
+ height = MIN(height, hints.max_height);
+ }
+ if (hints.flags & PMinSize) {
+ width = MAX(width, hints.min_width);
+ height = MAX(height, hints.min_height);
+ }
+
+ // Resize widget window
+ XResizeWindow(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ width, height);
+
+ // Get actual widget geometry
+ Window root;
+ int wx, wy;
+ unsigned int ww, wh;
+ unsigned int ignored;
+ XGetGeometry(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ &root,
+ &wx, &wy, &ww, &wh,
+ &ignored, &ignored);
+
+ // Center widget in allocation
+ wx = (allocation->width - ww) / 2;
+ wy = (allocation->height - wh) / 2;
+ XMoveWindow(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ wx, wy);
+ } else {
+ /* Child has not been realized, so unable to resize now.
+ Queue an idle resize. */
+ g_idle_add(idle_size_request, socket->plug);
+ }
+}
+
+static gboolean
+suil_x11_wrapper_key_event(GtkWidget* widget,
+ GdkEventKey* event)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(widget);
+
+ if (self->plug) {
+ return forward_key_event(self, (GdkEvent*)event);
+ }
+
+ return FALSE;
+}
+
+static void
+suil_x11_on_size_allocate(GtkWidget* widget,
+ GtkAllocation* a)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(widget);
+
+ if (self->plug
+ && GTK_WIDGET_REALIZED(widget)
+ && GTK_WIDGET_MAPPED(widget)
+ && GTK_WIDGET_VISIBLE(widget)) {
+ forward_size_request(self, a);
+ }
+}
+
+static void
+suil_x11_wrapper_class_init(SuilX11WrapperClass* klass)
+{
+ GObjectClass* const gobject_class = G_OBJECT_CLASS(klass);
+ GtkWidgetClass* const widget_class = GTK_WIDGET_CLASS(klass);
+
+ gobject_class->finalize = suil_x11_wrapper_finalize;
+ widget_class->realize = suil_x11_wrapper_realize;
+ widget_class->show = suil_x11_wrapper_show;
+ widget_class->key_press_event = suil_x11_wrapper_key_event;
+ widget_class->key_release_event = suil_x11_wrapper_key_event;
+}
+
+static void
+suil_x11_wrapper_init(SuilX11Wrapper* self)
+{
+ self->plug = GTK_PLUG(gtk_plug_new(0));
+ self->wrapper = NULL;
+ self->instance = NULL;
+ self->idle_iface = NULL;
+ self->idle_ms = 1000 / 30; // 30 Hz default
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ gtk_widget_set_size_request(GTK_WIDGET(handle), width, height);
+ return 0;
+}
+
+static gboolean
+suil_x11_wrapper_idle(void* data)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(data);
+
+ wrap->idle_iface->idle(wrap->instance->handle);
+
+ return TRUE; // Continue calling
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(wrapper->impl);
+
+ instance->host_widget = GTK_WIDGET(wrap);
+ wrap->wrapper = wrapper;
+ wrap->instance = instance;
+
+ const LV2UI_Idle_Interface* idle_iface = NULL;
+ if (instance->descriptor->extension_data) {
+ idle_iface = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ }
+ if (idle_iface) {
+ wrap->idle_iface = idle_iface;
+ wrap->idle_id = g_timeout_add(
+ wrap->idle_ms, suil_x11_wrapper_idle, wrap);
+ }
+
+ g_signal_connect(G_OBJECT(wrap),
+ "plug-removed",
+ G_CALLBACK(on_plug_removed),
+ NULL);
+
+ g_signal_connect(G_OBJECT(wrap),
+ "size-allocate",
+ G_CALLBACK(suil_x11_on_size_allocate),
+ NULL);
+
+ return 0;
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ if (wrapper->impl) {
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(wrapper->impl);
+ gtk_object_destroy(GTK_OBJECT(wrap));
+ }
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(
+ g_object_new(SUIL_TYPE_X11_WRAPPER, NULL));
+
+ wrapper->impl = wrap;
+ wrapper->resize.handle = wrap;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ gtk_widget_set_sensitive(GTK_WIDGET(wrap), TRUE);
+ gtk_widget_set_can_focus(GTK_WIDGET(wrap), TRUE);
+
+ const intptr_t parent_id = (intptr_t)gtk_plug_get_id(wrap->plug);
+ suil_add_feature(features, &n_features, LV2_UI__parent, (void*)parent_id);
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ // Scan for URID map and options
+ LV2_URID_Map* map = NULL;
+ LV2_Options_Option* options = NULL;
+ for (LV2_Feature** f = *features; *f && (!map || !options); ++f) {
+ if (!strcmp((*f)->URI, LV2_OPTIONS__options)) {
+ options = (LV2_Options_Option*)(*f)->data;
+ } else if (!strcmp((*f)->URI, LV2_URID__map)) {
+ map = (LV2_URID_Map*)(*f)->data;
+ }
+ }
+
+ if (map && options) {
+ // Set UI update rate if given
+ LV2_URID ui_updateRate = map->map(map->handle, LV2_UI__updateRate);
+ for (LV2_Options_Option* o = options; o->key; ++o) {
+ if (o->key == ui_updateRate) {
+ wrap->idle_ms = 1000.0f / *(const float*)o->value;
+ break;
+ }
+ }
+ }
+
+ return wrapper;
+}
diff --git a/src/x11_in_gtk3.c b/src/x11_in_gtk3.c
new file mode 100644
index 0000000..0c016c9
--- /dev/null
+++ b/src/x11_in_gtk3.c
@@ -0,0 +1,402 @@
+/*
+ Copyright 2011-2016 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <string.h>
+
+#include <gdk/gdkx.h>
+#include <gtk/gtk.h>
+#include <gtk/gtkx.h>
+
+#include "./suil_internal.h"
+
+#include "lv2/lv2plug.in/ns/ext/options/options.h"
+#include "lv2/lv2plug.in/ns/ext/urid/urid.h"
+
+#define SUIL_TYPE_X11_WRAPPER (suil_x11_wrapper_get_type())
+#define SUIL_X11_WRAPPER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), SUIL_TYPE_X11_WRAPPER, SuilX11Wrapper))
+
+typedef struct _SuilX11Wrapper SuilX11Wrapper;
+typedef struct _SuilX11WrapperClass SuilX11WrapperClass;
+
+struct _SuilX11Wrapper {
+ GtkSocket socket;
+ GtkPlug* plug;
+ SuilWrapper* wrapper;
+ SuilInstance* instance;
+ const LV2UI_Idle_Interface* idle_iface;
+ guint idle_id;
+ guint idle_ms;
+};
+
+struct _SuilX11WrapperClass {
+ GtkSocketClass parent_class;
+};
+
+GType suil_x11_wrapper_get_type(void); // Accessor for SUIL_TYPE_X11_WRAPPER
+
+G_DEFINE_TYPE(SuilX11Wrapper, suil_x11_wrapper, GTK_TYPE_SOCKET)
+
+/**
+ Check if 'swallowed' subwindow is known to the X server.
+
+ Gdk/GTK can mark the window as realized, mapped and visible even though
+ there is no window-ID on the X server for it yet. Then,
+ suil_x11_on_size_allocate() will cause a "BadWinow" X error.
+*/
+static bool
+x_window_is_valid(SuilX11Wrapper* socket)
+{
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(socket->plug));
+ Window root = 0;
+ Window parent = 0;
+ Window* children = NULL;
+ unsigned childcount = 0;
+
+ XQueryTree(GDK_WINDOW_XDISPLAY(window),
+ GDK_WINDOW_XID(window),
+ &root, &parent, &children, &childcount);
+ for (unsigned i = 0; i < childcount; ++i) {
+ if (children[i] == (Window)socket->instance->ui_widget) {
+ XFree(children);
+ return true;
+ }
+ }
+ XFree(children);
+ return false;
+}
+
+static gboolean
+on_plug_removed(GtkSocket* sock, gpointer data)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(sock);
+
+ if (self->idle_id) {
+ g_source_remove(self->idle_id);
+ self->idle_id = 0;
+ }
+
+ if (self->instance->handle) {
+ self->instance->descriptor->cleanup(self->instance->handle);
+ self->instance->handle = NULL;
+ }
+
+ self->plug = NULL;
+ return TRUE;
+}
+
+static void
+suil_x11_wrapper_finalize(GObject* gobject)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(gobject);
+
+ self->wrapper->impl = NULL;
+
+ G_OBJECT_CLASS(suil_x11_wrapper_parent_class)->finalize(gobject);
+}
+
+static void
+suil_x11_wrapper_realize(GtkWidget* w)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(w);
+ GtkSocket* const socket = GTK_SOCKET(w);
+
+ if (GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->realize) {
+ GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->realize(w);
+ }
+
+ gtk_socket_add_id(socket, gtk_plug_get_id(wrap->plug));
+
+ gtk_widget_realize(GTK_WIDGET(wrap->plug));
+
+ gtk_widget_set_sensitive(GTK_WIDGET(wrap->plug), TRUE);
+ gtk_widget_set_can_focus(GTK_WIDGET(wrap->plug), TRUE);
+ gtk_widget_grab_focus(GTK_WIDGET(wrap->plug));
+}
+
+static void
+suil_x11_wrapper_show(GtkWidget* w)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(w);
+
+ if (GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->show) {
+ GTK_WIDGET_CLASS(suil_x11_wrapper_parent_class)->show(w);
+ }
+
+ gtk_widget_show(GTK_WIDGET(wrap->plug));
+}
+
+static gboolean
+forward_key_event(SuilX11Wrapper* socket,
+ GdkEvent* gdk_event)
+{
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(socket->plug));
+ GdkScreen* screen = gdk_visual_get_screen(gdk_window_get_visual(window));
+
+ Window target_window;
+ if (gdk_event->any.window == window) {
+ // Event sent up to the plug window, forward it up to the parent
+ GtkWidget* widget = GTK_WIDGET(socket->instance->host_widget);
+ GdkWindow* parent = gtk_widget_get_parent_window(widget);
+ if (parent) {
+ target_window = GDK_WINDOW_XID(parent);
+ } else {
+ return FALSE; // Wrapper is a top-level window, do nothing
+ }
+ } else {
+ // Event sent anywhere else, send to the plugin
+ target_window = (Window)socket->instance->ui_widget;
+ }
+
+ XKeyEvent xev;
+ memset(&xev, 0, sizeof(xev));
+ xev.type = (gdk_event->type == GDK_KEY_PRESS) ? KeyPress : KeyRelease;
+ xev.root = GDK_WINDOW_XID(gdk_screen_get_root_window(screen));
+ xev.window = target_window;
+ xev.subwindow = None;
+ xev.time = gdk_event->key.time;
+ xev.state = gdk_event->key.state;
+ xev.keycode = gdk_event->key.hardware_keycode;
+
+ XSendEvent(GDK_WINDOW_XDISPLAY(window),
+ target_window,
+ False,
+ NoEventMask,
+ (XEvent*)&xev);
+
+ return (gdk_event->any.window != window);
+}
+
+static gboolean
+idle_size_request(gpointer user_data)
+{
+ GtkWidget* w = GTK_WIDGET(user_data);
+ gtk_widget_queue_resize(w);
+ return FALSE;
+}
+
+static void
+forward_size_request(SuilX11Wrapper* socket,
+ GtkAllocation* allocation)
+{
+ GdkWindow* window = gtk_widget_get_window(GTK_WIDGET(socket->plug));
+ if (x_window_is_valid(socket)) {
+ // Calculate allocation size constrained to X11 limits for widget
+ int width = allocation->width;
+ int height = allocation->height;
+ XSizeHints hints;
+ memset(&hints, 0, sizeof(hints));
+ XGetNormalHints(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ &hints);
+ if (hints.flags & PMaxSize) {
+ width = MIN(width, hints.max_width);
+ height = MIN(height, hints.max_height);
+ }
+ if (hints.flags & PMinSize) {
+ width = MAX(width, hints.min_width);
+ height = MAX(height, hints.min_height);
+ }
+
+ // Resize widget window
+ XResizeWindow(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ width, height);
+
+ // Get actual widget geometry
+ Window root;
+ int wx, wy;
+ unsigned int ww, wh;
+ unsigned int ignored;
+ XGetGeometry(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ &root,
+ &wx, &wy, &ww, &wh,
+ &ignored, &ignored);
+
+ // Center widget in allocation
+ wx = (allocation->width - ww) / 2;
+ wy = (allocation->height - wh) / 2;
+ XMoveWindow(GDK_WINDOW_XDISPLAY(window),
+ (Window)socket->instance->ui_widget,
+ wx, wy);
+ } else {
+ /* Child has not been realized, so unable to resize now.
+ Queue an idle resize. */
+ g_idle_add(idle_size_request, socket->plug);
+ }
+}
+
+static gboolean
+suil_x11_wrapper_key_event(GtkWidget* widget,
+ GdkEventKey* event)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(widget);
+
+ if (self->plug) {
+ return forward_key_event(self, (GdkEvent*)event);
+ }
+
+ return FALSE;
+}
+
+static void
+suil_x11_on_size_allocate(GtkWidget* widget,
+ GtkAllocation* a)
+{
+ SuilX11Wrapper* const self = SUIL_X11_WRAPPER(widget);
+
+ if (self->plug
+ && gtk_widget_get_realized(widget)
+ && gtk_widget_get_mapped(widget)
+ && gtk_widget_get_visible(widget)) {
+ forward_size_request(self, a);
+ }
+}
+
+static void
+suil_x11_wrapper_class_init(SuilX11WrapperClass* klass)
+{
+ GObjectClass* const gobject_class = G_OBJECT_CLASS(klass);
+ GtkWidgetClass* const widget_class = GTK_WIDGET_CLASS(klass);
+
+ gobject_class->finalize = suil_x11_wrapper_finalize;
+ widget_class->realize = suil_x11_wrapper_realize;
+ widget_class->show = suil_x11_wrapper_show;
+ widget_class->key_press_event = suil_x11_wrapper_key_event;
+ widget_class->key_release_event = suil_x11_wrapper_key_event;
+}
+
+static void
+suil_x11_wrapper_init(SuilX11Wrapper* self)
+{
+ self->plug = GTK_PLUG(gtk_plug_new(0));
+ self->wrapper = NULL;
+ self->instance = NULL;
+ self->idle_iface = NULL;
+ self->idle_ms = 1000 / 30; // 30 Hz default
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ gtk_widget_set_size_request(GTK_WIDGET(handle), width, height);
+ return 0;
+}
+
+static gboolean
+suil_x11_wrapper_idle(void* data)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(data);
+
+ wrap->idle_iface->idle(wrap->instance->handle);
+
+ return TRUE; // Continue calling
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(wrapper->impl);
+
+ instance->host_widget = GTK_WIDGET(wrap);
+ wrap->wrapper = wrapper;
+ wrap->instance = instance;
+
+ const LV2UI_Idle_Interface* idle_iface = NULL;
+ if (instance->descriptor->extension_data) {
+ idle_iface = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ }
+ if (idle_iface) {
+ wrap->idle_iface = idle_iface;
+ wrap->idle_id = g_timeout_add(
+ wrap->idle_ms, suil_x11_wrapper_idle, wrap);
+ }
+
+ g_signal_connect(G_OBJECT(wrap),
+ "plug-removed",
+ G_CALLBACK(on_plug_removed),
+ NULL);
+
+ g_signal_connect(G_OBJECT(wrap),
+ "size-allocate",
+ G_CALLBACK(suil_x11_on_size_allocate),
+ NULL);
+
+ return 0;
+}
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ if (wrapper->impl) {
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(wrapper->impl);
+ gtk_widget_destroy(GTK_WIDGET(wrap));
+ }
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(
+ g_object_new(SUIL_TYPE_X11_WRAPPER, NULL));
+
+ wrapper->impl = wrap;
+ wrapper->resize.handle = wrap;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ gtk_widget_set_sensitive(GTK_WIDGET(wrap), TRUE);
+ gtk_widget_set_can_focus(GTK_WIDGET(wrap), TRUE);
+
+ const intptr_t parent_id = (intptr_t)gtk_plug_get_id(wrap->plug);
+ suil_add_feature(features, &n_features, LV2_UI__parent, (void*)parent_id);
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ // Scan for URID map and options
+ LV2_URID_Map* map = NULL;
+ LV2_Options_Option* options = NULL;
+ for (LV2_Feature** f = *features; *f && (!map || !options); ++f) {
+ if (!strcmp((*f)->URI, LV2_OPTIONS__options)) {
+ options = (LV2_Options_Option*)(*f)->data;
+ } else if (!strcmp((*f)->URI, LV2_URID__map)) {
+ map = (LV2_URID_Map*)(*f)->data;
+ }
+ }
+
+ if (map && options) {
+ // Set UI update rate if given
+ LV2_URID ui_updateRate = map->map(map->handle, LV2_UI__updateRate);
+ for (LV2_Options_Option* o = options; o->key; ++o) {
+ if (o->key == ui_updateRate) {
+ wrap->idle_ms = 1000.0f / *(const float*)o->value;
+ break;
+ }
+ }
+ }
+
+ return wrapper;
+}
diff --git a/src/x11_in_qt4.cpp b/src/x11_in_qt4.cpp
new file mode 100644
index 0000000..c21357d
--- /dev/null
+++ b/src/x11_in_qt4.cpp
@@ -0,0 +1,151 @@
+/*
+ Copyright 2011-2015 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <QX11EmbedContainer>
+#include <QtEvents>
+#undef signals
+
+#include "./suil_config.h"
+#include "./suil_internal.h"
+
+extern "C" {
+
+typedef struct {
+ QX11EmbedContainer* host_widget;
+ QX11EmbedWidget* parent;
+} SuilX11InQt4Wrapper;
+
+class SuilQX11Container : public QX11EmbedContainer
+{
+public:
+ SuilQX11Container(SuilInstance* instance,
+ const LV2UI_Idle_Interface* idle_iface,
+ QX11EmbedWidget* widget)
+ : QX11EmbedContainer()
+ , _instance(instance)
+ , _idle_iface(idle_iface)
+ , _widget(widget)
+ , _ui_timer(0)
+ {}
+
+protected:
+ void showEvent(QShowEvent* event) {
+ if (_idle_iface && _ui_timer == 0) {
+ _ui_timer = this->startTimer(30);
+ _widget->embedInto(winId());
+ resize(_widget->size());
+ }
+ QX11EmbedContainer::showEvent(event);
+ }
+
+ void timerEvent(QTimerEvent* event) {
+ if (event->timerId() == _ui_timer && _idle_iface) {
+ _idle_iface->idle(_instance->handle);
+ }
+ QX11EmbedContainer::timerEvent(event);
+ }
+
+ void closeEvent(QCloseEvent* event) {
+ if (_ui_timer && _idle_iface) {
+ this->killTimer(_ui_timer);
+ _ui_timer = 0;
+ }
+ QX11EmbedContainer::closeEvent(event);
+ }
+
+private:
+ SuilInstance* const _instance;
+ const LV2UI_Idle_Interface* const _idle_iface;
+ QX11EmbedWidget* const _widget;
+ int _ui_timer;
+};
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ SuilX11InQt4Wrapper* impl = (SuilX11InQt4Wrapper*)wrapper->impl;
+
+ if (impl->parent) {
+ delete impl->parent;
+ }
+
+ if (impl->host_widget) {
+ delete impl->host_widget;
+ }
+
+ free(impl);
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ const LV2UI_Idle_Interface* idle_iface = NULL;
+ if (instance->descriptor->extension_data) {
+ idle_iface = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ }
+
+ SuilX11InQt4Wrapper* const impl = (SuilX11InQt4Wrapper*)wrapper->impl;
+ QX11EmbedWidget* const ew = impl->parent;
+
+ impl->host_widget = new SuilQX11Container(instance, idle_iface, ew);
+
+ instance->host_widget = impl->host_widget;
+
+ return 0;
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ QX11EmbedWidget* const ew = (QX11EmbedWidget*)handle;
+ ew->resize(width, height);
+ return 0;
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ SuilX11InQt4Wrapper* const impl = (SuilX11InQt4Wrapper*)
+ calloc(1, sizeof(SuilX11InQt4Wrapper));
+
+ SuilWrapper* wrapper = (SuilWrapper*)calloc(1, sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ QX11EmbedWidget* const ew = new QX11EmbedWidget();
+
+ impl->parent = ew;
+
+ wrapper->impl = impl;
+ wrapper->resize.handle = ew;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ const intptr_t parent_id = (intptr_t)ew->winId();
+ suil_add_feature(features, &n_features, LV2_UI__parent, (void*)parent_id);
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/src/x11_in_qt5.cpp b/src/x11_in_qt5.cpp
new file mode 100644
index 0000000..be90323
--- /dev/null
+++ b/src/x11_in_qt5.cpp
@@ -0,0 +1,148 @@
+/*
+ Copyright 2011-2015 David Robillard <http://drobilla.net>
+ Copyright 2015 Rui Nuno Capela <rncbc@rncbc.org>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#include <QWidget>
+
+#include <QTimerEvent>
+#include <QCloseEvent>
+
+#undef signals
+
+#include "./suil_config.h"
+#include "./suil_internal.h"
+
+extern "C" {
+
+typedef struct {
+ QWidget* host_widget;
+ QWidget* parent;
+} SuilX11InQt5Wrapper;
+
+class SuilQX11Widget : public QWidget
+{
+public:
+ SuilQX11Widget(QWidget* parent, Qt::WindowFlags wflags)
+ : QWidget(parent, wflags)
+ , _instance(NULL)
+ , _idle_iface(NULL)
+ , _ui_timer(0)
+ {}
+
+ void start_idle(SuilInstance* instance,
+ const LV2UI_Idle_Interface* idle_iface) {
+ _instance = instance;
+ _idle_iface = idle_iface;
+ if (_idle_iface && _ui_timer == 0) {
+ _ui_timer = this->startTimer(30);
+ }
+ }
+
+protected:
+ void timerEvent(QTimerEvent* event) {
+ if (event->timerId() == _ui_timer && _idle_iface) {
+ _idle_iface->idle(_instance->handle);
+ }
+ QWidget::timerEvent(event);
+ }
+
+ void closeEvent(QCloseEvent* event) {
+ if (_ui_timer && _idle_iface) {
+ this->killTimer(_ui_timer);
+ _ui_timer = 0;
+ }
+ QWidget::closeEvent(event);
+ }
+
+private:
+ SuilInstance* _instance;
+ const LV2UI_Idle_Interface* _idle_iface;
+ int _ui_timer;
+};
+
+static void
+wrapper_free(SuilWrapper* wrapper)
+{
+ SuilX11InQt5Wrapper* impl = (SuilX11InQt5Wrapper*)wrapper->impl;
+
+ if (impl->host_widget) {
+ delete impl->host_widget;
+ }
+
+ free(impl);
+}
+
+static int
+wrapper_wrap(SuilWrapper* wrapper,
+ SuilInstance* instance)
+{
+ SuilX11InQt5Wrapper* const impl = (SuilX11InQt5Wrapper*)wrapper->impl;
+ SuilQX11Widget* const ew = (SuilQX11Widget*)impl->parent;
+
+ if (instance->descriptor->extension_data) {
+ const LV2UI_Idle_Interface* idle_iface
+ = (const LV2UI_Idle_Interface*)
+ instance->descriptor->extension_data(LV2_UI__idleInterface);
+ ew->start_idle(instance, idle_iface);
+ }
+
+ impl->host_widget = ew;
+
+ instance->host_widget = impl->host_widget;
+
+ return 0;
+}
+
+static int
+wrapper_resize(LV2UI_Feature_Handle handle, int width, int height)
+{
+ QWidget* const ew = (QWidget*)handle;
+ ew->resize(width, height);
+ return 0;
+}
+
+SUIL_LIB_EXPORT
+SuilWrapper*
+suil_wrapper_new(SuilHost* host,
+ const char* host_type_uri,
+ const char* ui_type_uri,
+ LV2_Feature*** features,
+ unsigned n_features)
+{
+ SuilX11InQt5Wrapper* const impl = (SuilX11InQt5Wrapper*)
+ calloc(1, sizeof(SuilX11InQt5Wrapper));
+
+ SuilWrapper* wrapper = (SuilWrapper*)malloc(sizeof(SuilWrapper));
+ wrapper->wrap = wrapper_wrap;
+ wrapper->free = wrapper_free;
+
+ QWidget* const ew = new SuilQX11Widget(NULL, Qt::Window);
+
+ impl->parent = ew;
+
+ wrapper->impl = impl;
+ wrapper->resize.handle = ew;
+ wrapper->resize.ui_resize = wrapper_resize;
+
+ void* parent_id = (void*)(intptr_t)ew->winId();
+ suil_add_feature(features, &n_features, LV2_UI__parent, parent_id);
+ suil_add_feature(features, &n_features, LV2_UI__resize, &wrapper->resize);
+ suil_add_feature(features, &n_features, LV2_UI__idleInterface, NULL);
+
+ return wrapper;
+}
+
+} // extern "C"
diff --git a/suil.pc.in b/suil.pc.in
new file mode 100644
index 0000000..1bff552
--- /dev/null
+++ b/suil.pc.in
@@ -0,0 +1,10 @@
+prefix=@PREFIX@
+exec_prefix=@EXEC_PREFIX@
+libdir=@LIBDIR@
+includedir=@INCLUDEDIR@
+
+Name: Suil
+Version: @SUIL_VERSION@
+Description: LV2 plugin UI hosting library
+Libs: -L${libdir} -l@LIB_SUIL@
+Cflags: -I${includedir}/suil-@SUIL_MAJOR_VERSION@
diff --git a/suil/suil.h b/suil/suil.h
new file mode 100644
index 0000000..5f1b7fe
--- /dev/null
+++ b/suil/suil.h
@@ -0,0 +1,300 @@
+/*
+ Copyright 2011-2017 David Robillard <http://drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+/**
+ @file suil.h API for Suil, an LV2 UI wrapper library.
+*/
+
+#ifndef SUIL_SUIL_H
+#define SUIL_SUIL_H
+
+#include <stdint.h>
+
+#include "lv2/lv2plug.in/ns/lv2core/lv2.h"
+
+#ifdef _WIN32
+# define SUIL_LIB_IMPORT __declspec(dllimport)
+# define SUIL_LIB_EXPORT __declspec(dllexport)
+#else
+# define SUIL_LIB_IMPORT __attribute__((visibility("default")))
+# define SUIL_LIB_EXPORT __attribute__((visibility("default")))
+#endif
+
+#ifdef SUIL_SHARED
+# ifdef SUIL_INTERNAL
+# define SUIL_API SUIL_LIB_EXPORT
+# else
+# define SUIL_API SUIL_LIB_IMPORT
+# endif
+#else
+# define SUIL_API
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#else
+# include <stdbool.h>
+#endif
+
+/**
+ @defgroup suil Suil
+
+ Suil is a library for loading and wrapping LV2 plugin UIs. With Suil, a
+ host written in one supported toolkit can embed a plugin UI written in a
+ different supported toolkit. Suil insulates hosts from toolkit libraries
+ used by plugin UIs. For example, a Gtk host can embed a Qt UI without
+ linking against Qt at compile time.
+
+ Visit <http://drobilla.net/software/suil> for more information.
+
+ @{
+*/
+
+/**
+ UI host descriptor.
+
+ This contains the various functions that a plugin UI may use to communicate
+ with the plugin. It is passed to suil_instance_new() to provide these
+ functions to the UI.
+*/
+typedef struct SuilHostImpl SuilHost;
+
+/** An instance of an LV2 plugin UI. */
+typedef struct SuilInstanceImpl SuilInstance;
+
+/** Opaque pointer to a UI handle. */
+typedef void* SuilHandle;
+
+/** Opaque pointer to a UI widget. */
+typedef void* SuilWidget;
+
+/**
+ UI controller.
+
+ This is an opaque pointer passed by the user which is passed to the various
+ UI control functions (e.g. SuilPortWriteFunc). It is typically used to pass
+ a pointer to some controller object the host uses to communicate with
+ plugins.
+*/
+typedef void* SuilController;
+
+/** Function to write/send a value to a port. */
+typedef void (*SuilPortWriteFunc)(
+ SuilController controller,
+ uint32_t port_index,
+ uint32_t buffer_size,
+ uint32_t protocol,
+ void const* buffer);
+
+/** Function to return the index for a port by symbol. */
+typedef uint32_t (*SuilPortIndexFunc)(
+ SuilController controller,
+ const char* port_symbol);
+
+/** Function to subscribe to notifications for a port. */
+typedef uint32_t (*SuilPortSubscribeFunc)(
+ SuilController controller,
+ uint32_t port_index,
+ uint32_t protocol,
+ const LV2_Feature* const* features);
+
+/** Function to unsubscribe from notifications for a port. */
+typedef uint32_t (*SuilPortUnsubscribeFunc)(
+ SuilController controller,
+ uint32_t port_index,
+ uint32_t protocol,
+ const LV2_Feature* const* features);
+
+/** Function called when a control is grabbed or released. */
+typedef void (*SuilTouchFunc)(
+ SuilController controller,
+ uint32_t port_index,
+ bool grabbed);
+
+/** Initialization argument. */
+typedef enum {
+ SUIL_ARG_NONE
+} SuilArg;
+
+/**
+ Initialize suil.
+
+ This function should be called as early as possible, before any other GUI
+ toolkit functions. The variable argument list is a sequence of SuilArg keys
+ and corresponding value pairs for passing any necessary platform-specific
+ information. It must be terminated with SUIL_ARG_NONE.
+*/
+SUIL_API
+void
+suil_init(int* argc, char*** argv, SuilArg key, ...);
+
+/**
+ Create a new UI host descriptor.
+ @param write_func Function to send a value to a plugin port.
+ @param index_func Function to get the index for a port by symbol.
+ @param subscribe_func Function to subscribe to port updates.
+ @param unsubscribe_func Function to unsubscribe from port updates.
+*/
+SUIL_API
+SuilHost*
+suil_host_new(SuilPortWriteFunc write_func,
+ SuilPortIndexFunc index_func,
+ SuilPortSubscribeFunc subscribe_func,
+ SuilPortUnsubscribeFunc unsubscribe_func);
+
+/**
+ Set a touch function for a host descriptor.
+
+ Note this function will only be called if the UI supports it.
+*/
+SUIL_API
+void
+suil_host_set_touch_func(SuilHost* host,
+ SuilTouchFunc touch_func);
+
+/**
+ Free `host`.
+*/
+SUIL_API
+void
+suil_host_free(SuilHost* host);
+
+/**
+ Check if suil can wrap a UI type.
+ @param host_type_uri The URI of the desired widget type of the host,
+ corresponding to the `type_uri` parameter of suil_instance_new().
+ @param ui_type_uri The URI of the UI widget type.
+ @return 0 if wrapping is unsupported, otherwise the quality of the wrapping
+ where 1 is the highest quality (direct native embedding with no wrapping)
+ and increasing values are of a progressively lower quality and/or stability.
+*/
+SUIL_API
+unsigned
+suil_ui_supported(const char* host_type_uri,
+ const char* ui_type_uri);
+
+/**
+ Instantiate a UI for an LV2 plugin.
+
+ This funcion may load a suil module to adapt the UI to the desired toolkit.
+ Suil is configured at compile time to load modules from the appropriate
+ place, but this can be changed at run-time via the environment variable
+ SUIL_MODULE_DIR. This makes it possible to bundle suil with an application.
+
+ Note that some situations (Gtk in Qt, Windows in Gtk) require a parent
+ container to be passed as a feature with URI LV2_UI__parent
+ (http://lv2plug.in/ns/extensions/ui#ui) in order to work correctly. The
+ data must point to a single child container of the host widget set.
+
+ @param host Host descriptor.
+ @param controller Opaque host controller pointer.
+ @param container_type_uri URI of the desired host container widget type.
+ @param plugin_uri URI of the plugin to instantiate this UI for.
+ @param ui_uri URI of the specifically desired UI.
+ @param ui_type_uri URI of the actual UI widget type.
+ @param ui_bundle_path Path of the UI bundle.
+ @param ui_binary_path Path of the UI binary.
+ @param features NULL-terminated array of supported features, or NULL.
+ @return A new UI instance, or NULL if instantiation failed.
+*/
+SUIL_API
+SuilInstance*
+suil_instance_new(SuilHost* host,
+ SuilController controller,
+ const char* container_type_uri,
+ const char* plugin_uri,
+ const char* ui_uri,
+ const char* ui_type_uri,
+ const char* ui_bundle_path,
+ const char* ui_binary_path,
+ const LV2_Feature* const* features);
+
+/**
+ Free a plugin UI instance.
+
+ The caller must ensure all references to the UI have been dropped before
+ calling this function (e.g. it has been removed from its parent).
+*/
+SUIL_API
+void
+suil_instance_free(SuilInstance* instance);
+
+/**
+ Get the handle for a UI instance.
+
+ Returns the handle to the UI instance. The returned handle has opaque type
+ to insulate the Suil API from LV2 extensions, but in pactice it is currently
+ of type `LV2UI_Handle`. This should not normally be needed.
+
+ The returned handle is shared and must not be deleted.
+*/
+SUIL_API
+SuilHandle
+suil_instance_get_handle(SuilInstance* instance);
+
+/**
+ Get the widget for a UI instance.
+
+ Returns an opaque pointer to a widget, the type of which matches the
+ `container_type_uri` parameter of suil_instance_new(). Note this may be a
+ wrapper widget created by Suil, and not necessarily the widget directly
+ implemented by the UI.
+*/
+SUIL_API
+SuilWidget
+suil_instance_get_widget(SuilInstance* instance);
+
+/**
+ Notify the UI about a change in a plugin port.
+ @param instance UI instance.
+ @param port_index Index of the port which has changed.
+ @param buffer_size Size of `buffer` in bytes.
+ @param format Format of `buffer` (mapped URI, or 0 for float).
+ @param buffer Change data, e.g. the new port value.
+
+ This function can be used to notify the UI about any port change, but in the
+ simplest case is used to set the value of lv2:ControlPort ports. For
+ simplicity, this is a special case where `format` is 0, `buffer_size` is 4,
+ and `buffer` should point to a single float.
+
+ The `buffer` must be valid only for the duration of this call, the UI must
+ not keep a reference to it.
+*/
+SUIL_API
+void
+suil_instance_port_event(SuilInstance* instance,
+ uint32_t port_index,
+ uint32_t buffer_size,
+ uint32_t format,
+ const void* buffer);
+
+/**
+ Return a data structure defined by some LV2 extension URI.
+*/
+SUIL_API
+const void*
+suil_instance_extension_data(SuilInstance* instance,
+ const char* uri);
+
+/**
+ @}
+*/
+
+#ifdef __cplusplus
+} /* extern "C" */
+#endif
+
+#endif /* SUIL_SUIL_H */
diff --git a/waf b/waf
index e22930a..9c57f6f 100755
--- a/waf
+++ b/waf
@@ -1,16 +1,169 @@
#!/usr/bin/env python
+# encoding: latin-1
+# Thomas Nagy, 2005-2017
+#
+"""
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions
+are met:
-# Minimal waf script for projects that include waflib directly
+1. Redistributions of source code must retain the above copyright
+ notice, this list of conditions and the following disclaimer.
-from waflib import Context, Scripting
+2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in the
+ documentation and/or other materials provided with the distribution.
-import inspect
-import os
+3. The name of the author may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
-def main():
- script_path = os.path.abspath(inspect.getfile(inspect.getmodule(main)))
- project_path = os.path.dirname(script_path)
- Scripting.waf_entry_point(os.getcwd(), Context.WAFVERSION, project_path)
+THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
+IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
+INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
+IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGE.
+"""
+
+import os, sys, inspect
+
+VERSION="2.0.4"
+REVISION="5e4201f8b4f4a2a744d52154db826b02"
+GIT="x"
+INSTALL=''
+C1='#<'
+C2='#4'
+C3='#)'
+cwd = os.getcwd()
+join = os.path.join
+
+
+WAF='waf'
+def b(x):
+ return x
+if sys.hexversion>0x300000f:
+ WAF='waf3'
+ def b(x):
+ return x.encode()
+
+def err(m):
+ print(('\033[91mError: %s\033[0m' % m))
+ sys.exit(1)
+
+def unpack_wafdir(dir, src):
+ f = open(src,'rb')
+ c = 'corrupt archive (%d)'
+ while 1:
+ line = f.readline()
+ if not line: err('run waf-light from a folder containing waflib')
+ if line == b('#==>\n'):
+ txt = f.readline()
+ if not txt: err(c % 1)
+ if f.readline() != b('#<==\n'): err(c % 2)
+ break
+ if not txt: err(c % 3)
+ txt = txt[1:-1].replace(b(C1), b('\n')).replace(b(C2), b('\r')).replace(b(C3), b('\x00'))
+
+ import shutil, tarfile
+ try: shutil.rmtree(dir)
+ except OSError: pass
+ try:
+ for x in ('Tools', 'extras'):
+ os.makedirs(join(dir, 'waflib', x))
+ except OSError:
+ err("Cannot unpack waf lib into %s\nMove waf in a writable directory" % dir)
+
+ os.chdir(dir)
+ tmp = 't.bz2'
+ t = open(tmp,'wb')
+ try: t.write(txt)
+ finally: t.close()
+
+ try:
+ t = tarfile.open(tmp)
+ except:
+ try:
+ os.system('bunzip2 t.bz2')
+ t = tarfile.open('t')
+ tmp = 't'
+ except:
+ os.chdir(cwd)
+ try: shutil.rmtree(dir)
+ except OSError: pass
+ err("Waf cannot be unpacked, check that bzip2 support is present")
+
+ try:
+ for x in t: t.extract(x)
+ finally:
+ t.close()
+
+ for x in ('Tools', 'extras'):
+ os.chmod(join('waflib',x), 493)
+
+ if sys.hexversion<0x300000f:
+ sys.path = [join(dir, 'waflib')] + sys.path
+ import fixpy2
+ fixpy2.fixdir(dir)
+
+ os.remove(tmp)
+ os.chdir(cwd)
+
+ try: dir = unicode(dir, 'mbcs')
+ except: pass
+ try:
+ from ctypes import windll
+ windll.kernel32.SetFileAttributesW(dir, 2)
+ except:
+ pass
+
+def test(dir):
+ try:
+ os.stat(join(dir, 'waflib'))
+ return os.path.abspath(dir)
+ except OSError:
+ pass
+
+def find_lib():
+ src = os.path.abspath(inspect.getfile(inspect.getmodule(err)))
+ base, name = os.path.split(src)
+
+ #devs use $WAFDIR
+ w=test(os.environ.get('WAFDIR', ''))
+ if w: return w
+
+ #waf-light
+ if name.endswith('waf-light'):
+ w = test(base)
+ if w: return w
+ err('waf-light requires waflib -> export WAFDIR=/folder')
+
+ dirname = '%s-%s-%s' % (WAF, VERSION, REVISION)
+ for i in (INSTALL,'/usr','/usr/local','/opt'):
+ w = test(i + '/lib/' + dirname)
+ if w: return w
+
+ #waf-local
+ dir = join(base, (sys.platform != 'win32' and '.' or '') + dirname)
+ w = test(dir)
+ if w: return w
+
+ #unpack
+ unpack_wafdir(dir, src)
+ return dir
+
+wafdir = find_lib()
+sys.path.insert(0, wafdir)
if __name__ == '__main__':
- main()
+
+ from waflib import Scripting
+ Scripting.waf_entry_point(cwd, VERSION, wafdir)
+
+#==>
+#BZh91AY&SY¿øGH¢«ÿÿ°ÐÿÿÿÿÿÿÿÿÿÿÿE ‚„ 0Á#)€¨bIwzØ8#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)#)_mßmêõ¶V3[Y¢T¬ªTª[+îç$é²ßs§a¡¼ÛZ¥Û7­Ä»}5öîÛÝ©ÔÙ¥­«{n&[>o{íoœ#4¼Í–§m‘‘@õÑ&ö÷vîØ-nÖjîÀr/wtvÎï}Ùß_2Hô#4ï›Ý<2uÔ;ÍsÃÝïízܳ]°dÛs·Þß=¯xn`n™ámöžë»ÕhìWmC%ïgÞ#)#)#)ì#)#)ìhÀ:W­p>Ž€§Ûƃë»;Íæ#)½·°éï6ŸG»Iæ(€Ž¹Æܯ¡§(¦¶#<hH4#4@6Å#)w³Þª©*‰*JzaŠ(‘ET*¤%JvÏN»Û{hÝöÇzu“³+ëîÏŒë»tÛ|÷¼÷ÜàUkPÛ¦¨®Ùw7W}æœÚ…d°#4¼ße÷jº}z˱뤽ӽ7½y_Y$æg·Ö×VöËÞû‰2:eVÞ}ž¼ËTß3µiwkÞú=o7l}=tºÍhÍ*HTJzz:»{›L‡½ì+{Þ»Üîî¼4÷†½iN—zttÃÛ[m *«Þv{ÚqN€Ð¨®Àty@#)":èT›½¹êŒÞå^»¨UÙ­o¸½ÝÖùœ}¾ûèè:}ó»nœQkh,²Ÿv/Œ@Ý®ûp«f6¾'yáèÓ×9½z»·Üîè6ã«îÍÞöz¾ÇÇ ö×–Ìö:ÓDI–*[ÙÏwo<sv´X–¼.óï­í±Öº»£¹®ç7³s7·½›Fìç=÷yÙîÖîÎíݳo¼=å7·ËÞ{×Ü­¬¯GÞ÷½ª×5ÕÄÙ¢6ힺˆË³s°[¹O ôµË’no‡®nÎíÍW5±>>yÞó8ºï{»zô{æï6z\XîY¥ìyöó‹>ãCÖœoq”àrhfíTêæñf#)»)绽=;—}Í=äy³­^:!B€ ¨ŠjR#<FÀ4U·:Ûw@æE6È+fݳî t®ÛvÁlOsºÙÑ^½^Ò;³ ]ww+ÍÃÞjª¦ë½Ý§;+&zço#)#)îóÕ®@#)»Ì½]ÎëçµÏ|í]ÎhÕ׬wk¹U£;g³¼ðî7«¤âŽ7:G´yÛ]#)}S—¾ÌKJªåµ¦&¸è‚öày¼Ùï±ñ7ßß^ó§ÀU£&Šão1éíhò6é£m¸t}°÷Ýϼ»“µƒŸmÄo«T&ùµ{¹ËI±qÚOxûÞÛ¸x¹£Û¯zøÞ½é¶ÑaÏwžúçÏ­¶° @KM#)>N›Ø÷·{w±÷šÛݱï7ní Ti}x|1çÛ-{¹Þ½STëœö]#)#<ÖzSkvh3´îͽuê¶=;u»Ò\aîôáSßM;ϼôzöà »ƒ]b¶ôÚAw>¾õî}Y½Û¢š=ôØõìë®éׯ:Ó¶o,7zÙêšÒ +;›=yxö÷×ÓГʹ4ôNÝÚRö»X/t¨=ñç[½3¼|æ{Þp»h @ÓgKxï{‰¶õ“‰âfu®.2ªÍ‚ô}ܯ^;¶Ž »uEíœéÊuª“Nî1÷ƒDçºíŪDzWn$š³ÙåÅ>¾÷>µŸ-õðƒœ¾§]`·Û´|Gf É£Ëë36ñÀ¦Î­{³¹ÝŽ{»›Ãt6ß'¾û½o¶¡´=ÝÛ†¾.»OSx¼¡–Ÿ|7#44@#)š#)   #)£@Ñ¡#4#Pž§¤ò€õ駩â‚S@„@ КjfhMQú=TýHÄÑ‘“Ò#)@#)#)#)#)$!#@L§¦¦Š~&¦hÈÔžS@=OB©å#)Ð#)#)#)#)$õII"i¤ÓSÓ ¡’ˆöOÕÔŒ€#)zš¡ Ñê#)#)#)#)$ˆ&€#)#)€š#)F¦£OS jbdi©é#4#4#4F€#)$ÔDÐLA22& =SÔzŸ¥<§éª~©è@ö¡#)#)#)#)ÿûð?ìÛV»‰¬åÐnVÕÜ—ùVœP¦"1¾ý«]Ò „`¶5²Ù—ÛU¶¹[Z¢Û$}ij B?3ôÿçuùÍfñÒnÌR’ÓyŽDR¹üâÓÍ;ÅbÄâä¦ÅBÅ]ÄÂDgû…àŠDªݠJ¿6(9ÅZt¹w®Kˆ\<´ÓÅÄÕ:§NªmæŠx³—ž8––«™¾ É6Þo\qMfû)‰#CýþSgÈ…¢)¬U"$F1PcRD ÁR -E‚ Ò‚Dj Q®S”°„‰T̬ S@* „Ž$QBÀ$PlDP„\A¢#)ÙQRjÕÝ©ÑlV-õwmI[jªßkmjµ33!d£3P)¦&ÃKI!4›™‰’Âj™c ¥(ÒŠm¤Æ š$ &Œ†©h¶5ÑšK&–F# %£EF#"m)FXÔ±¡LXÄÆ*P6LØÒE&ÉhÚˆ¡,²Ò˜ÊÑTD,´†ÍÁ²“5ÖEhÀ*#4&ÈŠ¤Ø¦2P&4Í2Ò1¥cR”l%´Ú¶U­ˆ„Õ Ø“FfIš"ƒ,›i¶ÓM MIFÊJšØËRÛ2¦ZL¦’ˆÑ¥„5fŠIK#<YXÅ2B$Ê‹4ŠÅ&ÅD‚6)5"QšTb#<”Æ(Ø„!A"2˜À´2#$©0d"Ë,Í DDŠ ¤šhÒ²€ÌÑ‘A"ˆ¥“Y(£6!…¤X†K) 5”¤„ÅRd`¥4(†bARhÒ%Òd#4d*1bH°LÊZb)˜ÆÙ!˜¢Æi¦l&hX©6&ÊÂ`D’“b±%£– $M¦(”#<ˆ˜e d”„bÍ ”JJƦj6+ i5$I1„´¥HȲ[E–e‹2MfR#c&ÄÈŠQ›"`™¶iŒ)`²Æ*CdƒM!£•’ÉÊš$RÉ! š"ÓMH4Z„$–e6Q”´S3HÚ#AÒÅÚ˜,Pi,ÆB1d¤²$˜‹Ó0¢”Q£0Á¨¦…&*#4‚$H$0ŠLÆ’Fˆ‘2Âb‚6I•35f4U)TJ"fCR4Ä4ÍŠ”6ÊhÙ#Eb’I6H£Q£%M‘iš,lE4™I™LRÈÊͦD¬D`‰)™#<ji4É!Œ¢4†¦‹M±DlÔ©L˜S#$dÉ)Y4XÉXŒeIÌEcI¢M%MFɨ²I41Á2Ù¤ÄÚ#4*`Ë0šL¢ˆ±¦d6dI)²S#J ›&BjÍ•‹XK‘1&™d‰,[F±¨5ˆ¦[#<D%ؤH0Œ£RX´i†S4Ò˜6JÊiLšÒA¡$Å)1˜B†Y6De0H¦Ê2ÂÒ ¡2&c"HÑ6Ú•[FPÉBZeŒ¦cf”RJf"+A©K`RÍb“Le e•6Y6 ‹S-…³F&¦Äe( –BR×÷úí²!55@ˆ# c6Š1Q­‚dš4š‹Y$RÊhĨhÖ‰¥¤*66RMFÖDe£Q–Le"Ê`%IŒm" 134¥³5ciP¶6¢´dÓ–2©$ÌŒÖ6CJÍÉ“-&ÐlYV6Å#EM”#k"e‘TƳQL)³c)36Ém6cFÖ‹DkdÓ5’J™e6²kE6ÊRšcf5©HÉHÚ‹¬±&“R&#<TFÑU‚µ´d¨‹X¶“Y(¶£h¨©5bh‰’*‹Q*Ñ°ËF¬Xƒi4”4BkPÂJ)Ú TLD-2LJÓXØ5LKcT–#4ÒAVÕ+mšiU ”ÍMY4I²1!š„QdŠÚ–³lb”Úm–iS¦›TÈ‹m*–I35J‘‹4Ñ ¶„ƒ3$Ëe ¦RK&ÙYRh–QŒ2ÂEE‰ 3 RbÅ…‚Aˆ‘"Aa1ˆl¶™ªf%E šÒVI¥“&Š6D6X¤Se hÙY©D,JfIaIE˜¦`bjZ5™!•Šm&J#È”„f™$„F£&CJ2‰ ”Ô@ÄÂÒcA’†Z,Q4¦E1fY¥l&1(,˜Å0”²L²Bj-"£LµQ³)$£ …Œ…Qe+#L6Š“’Œ›#RE¥„Š2V4‰#4C1“I¤Ù”Ù””³6”¤¬S†‹$j!I+EQ³Z3DV*SI°Ó,k2¨Ð$d# LR#416$©6“(QRšb¤‚Š[%2£h*™Sd²T™Bb¢M$€‰CJ2Vd–Ñ["6"™™"TA„†Ë1DC&ÄÍ#b5šMÌDؘ‰©! ´hÑl%EF¢ŒjÆe°ÒaŠPÈR4²dlb”Al ¨+,²Y …X´fJl¡*Hƒm QXÑ&‚R¨ÑA±A’6MQh4˜J!…„¥(jcL0ÙJ£TF”ÓK!E$ɶdZ$ Z‹±JªZhª-&‹%‰"še+(V, Ì #)˜™%M‰DLÒU¨¬b¦j*KEŒ)EšZŒ•‹[IŠJÍ*‘6Ñ´MbS!™MIa@Y,T”lš$ˆÄ•IHk™F-f@ÐCQe*eLE6Ô•’Ù ¤)¤šF#<&Íi6‹e$±RcI#F,$ŠÔ¢4m,´Z6±&ÖŠ´e(ÚJ©BÔm£RQb‘M ’S#0&4Ô‰(53d™µŠ¤±j’¥š²kBV€‹cRTQ$[b™¶imE±hÖ£bÛI™3e–[+mE™QµM•¥©E1©(22“S%cE‚$Ñ¢ŒM&kcS6$‹A€´•±dÑ´¦šËQXÖ3HÚ2Q¢Éckm–µŠDÍ¢TÄ”c ,EÒJÓ#%XÛØ´Í[F5¬´•42ÖÊY-©©¶BÚ5bhÊ(ˆ±š+$¢ƒa¬Ôš±™³eDPj#I”“"2(ŒT–Ò`,dÿ‡ýoûÿÍ@ÿÏ?ºxåk3HHÿ‡û1µ¦pš¦ÜtKG,#4“÷‚á„P)«I³åÕ×yo9¿Çþ>ßøÊnã•Qÿ+k™÷z#<œ0(!¯þªÊëQh0\:8À¥‚á‰I äˆm½µð8|{'ýlÄêÖÿ—çµ£þ²IB–¹@%m#<ÊH–á+„Oájq•™ÆÐd ¡§}Ó,uË$ɬ ÈÏu‹Ã4ëZvÄäðó"28”caªN ׋’Æ#ükfØíRRWü6S;qC-“P #´¼2&Ö8cDOL#nŽ#¢¢5uB*35S4™³(·\°ww,¼ëß±Ög5r#&ŠéP¦(M Ñ™N¸ÒæX”…0€Ø$ÄÓb\ÚìY#)ïcFq#)Ž¯=ݨe}íÍ‘›ãsI`²"ÄAL^.LUA­ì.9_ü6X± nᦠ¢Á`îÖõ@S#<-«‰~åÍ$?'VähÅ'ô«ß`v¼^˜­D˳b(¦¦Ð¤Y;ø䬭èI³7>#4´¡±;ž"J<våÅþÈÃÕ…wÓ#<ÐA¢#4#4e†íŸíÿ‡^½©êuQYÊ„ìÚmû»ÃÀöø°#4kÞÞîŠõç¯d£—.‹¸¦W6éd4QW6éîè±±ByFâIi‚ˆÈ¤÷Õ}HZ~¶¹Í}îEmåÍçHŸê·`+Ž5g=â aãÆa#4›O,¡´Ã”¶ ªÅaåtFaÑ#))‚4†ÓŒ®4Úï9äË FÔ£+ÜßesF âk®W{»£¥ræJ)uÝ$Žs!_±¶¬ºŸøõ–j˜“T¤R…",0%+¡‡vØ˦N#-X,úè¤OUM?6¦#4+j~/LHeàÁ¿ã©HlÉÏLª #Š‡û“TÄC Ã1|êbæ_Å-  ŒE±õôÄ9 Gášå®oýó„бÅT¶[ DûjºïrûDá÷Ô½ëçÏù¹blù^ýN·˜Wêcv\â|²~_Bå,rpÕ{q¦rugbhÀà&É/z®¸3,Ìk·–·‡¹^Mm^öÅ[)•TRŸ·_²ýNyßCµ;7ktä××P¦±´HNBÉF¢óe³¦µ¼”ROÖãܲa2³Ñ…’¡Øäm¬ÓM¨ÑZ+<œTê)AG9åxû_¯:+”‹¢jË*¨6~YjdÒo$~ ¡GÞ.Í¡uQF¨Í]ÿµS,b“ Cÿ“PV1 œh£ÊŠª‚Ä7ÌçáüY`ãPm±›qí…È<KB«å E|™›ÛJÄóeká]Y)¢©£d*(«ÍË&£yËQVÿ·ý]½1§éu&×Ò¹dÂ[ôÿ#<™¢(5A)AQUyký7!„dLcI©)>ç7+¾Î J†’RtNÜPL ªrºÊREÂPɢ幨¶Þïåß>o4¤–ÒT>×rÅ¢¿¶[¯³±ûzãH ¨ÎTi®%§´JÖª…¢Y£d«ñ~…y‹A¤·ëtÁH»Që½=¸¯-6¹€`|š˜¨R4Àå­Ib?¢¡ðçÔ°Ê#]+k6gÞ’|RšÏætþn¶ìÂm“_ι%ú* nÖí L+¯ ÃPHÁe!BRkïycÝ\œºas­~.Y#«~˜05TàǪ4ÆéE2„R]U¥"½›Õ¬üjç“¢oåYNNYlÓ¾¬ Bj/Âë¢ûÚû¥y¾ç-%»³¸^SÉhTþqF#4¥¶–š·›{à ­Â„S²ŠŸF¯’Ø­-ŠŒ:¤àÏ[0"<™% kTt©@'…SuïoÒ‚5Ú¶|åñ‡\i*Ï'üI½øÄ#V.(¢,œ½_Óg«<þÌBׂh—:p ±çTnÅIÑ~pϘ컈  ´ÅõæI['¯Ï5®U%&aFeS?™Áú0ú‰0#ùNã¨j# <0«ÃD?î3ÊùÑ}›€g¬ïSà ~$–ªÓ#rB‘ä…E8Õ*m]*w;=÷ág#<£õÉYJéã›÷ä f¢Õ€²·2½ýõWSn=|f\ˆ1¹Åšh–w°û²ÓdA³ÁjC?L[»ç&Cë|´Þ¾©œTbtNlžLÑÛå\Sƒ›ágS¹C­W™Brç|Pââ‰Í£ìe#<w§çÖúüN¦¢m‘0Ÿx¶ÌµèùÍBùgŠä¾UPµ:ƒÃ:ëƒÝéJ}9Ñ&ˆEÔý8²¬#=é#)à„ña¨âüÚÌ8`³h†÷ !dúè{‘š¥¿EvØI£(^ÑýQŒÇ\TŠ%ò¨ÇhÔ˜†t¾©Ò‡Î\_„Mz}ÜMïuXaãHt¡°îŒ"$˜I?5G…«˜& ÜŸj¤btÇdzï†s4çíßz`s¨UYLó¬YI×jµÙPÝ(Ib¨Œ1Gù|ÛýY<@¡&Ì\2wCK´ãÆfÁ‰#<B…!¬;?Ý;|œ–ÅÇ’'OO7WCåýQ8êÐ8…ƒw H”gÉÎSVïpàëîM×5²’ÝÍ•‰»!ŒÊâr(õøå1§ ²´ :+ho*„LüÓa7~·µÕHا.®Aù¬dsñ(>>3h—qÒI–ÇÊ!uïZèîÙ­<œ#<Î5DVhœ0VÊ”‹¾û\ð#4ô•½Ç­j¸7üÑPm½I¦ˆG G}ø;wB1´nn­:ýÇP§ ùq69û7O›#4sOl#<ߧ•òxì„’HeT×Ƕ–“çÚ”Ò£o¨#4_Ó“F÷0ÈiéAÝ| 9Ý#4ú!&u;.xöcVyaŸÞþ½iœa7'àBX£ñïO—QÎ÷T­½@[â\|ˆ]20Ó›‘ò#48@âV¶ã[ü(½öcm‡cë#;jÑ1ê-—ðübü¥õoÉæ1ýœÍ_7ï÷búò!ª‘Al¯ª©-öò®ßJáu«××¹€ RÔ(sd½ë¡)$©yoyÑ4þ.ä7+¯v™¸Íú:º•d ~¦A³5Ú•òçÉÆ´Ô²7"ËåÿŠ¢Â(O#<©¯ÿU×àþñ:–Ÿ/ÇÆñѧôUAûÚþµæ'ãvßÓ×yAb5+좙òg€ŸnožÏ®•Wsv±GrŸ²ZûßßÎ{bˆûfìd‘?ÒÖھ黄«4jMPk²éPC¢«55Á)™ïÅÌøÅ»?–‡mç-kY¥¬˜Â}ùkw¨i˜ûì«?™ØNÜieT¨0B„]~IvS<f}©Øè«4j=,jé.ì³åV\åRøž§$\œ%ûèF#4Öc;ý9Ñûôº'ò#4[ºÄ8G àDL4wwÄ9½ºÛ9ó¬ñùÀnÀêÀ¡´¥Uü“8÷fãéEEâf¥#46&Òu‡éõ´†Nîc"âSå¸hÆ~=Sïß¿|mŠ?A—Žýñ`ʼnÙýá¿i˜êì齇µ­Æý"ñç¹mw`aI-§=oƒQñ/r+ÕÎrðaYAx‡k>YàÉqb™1ôô†6ž«¤¾=GØDM×~”¶÷?èšR|¹Ã‹Þð¾ûçÛaïDC,½ïZF™¦"&gËŒ¦_gð¢åT¼™µÉOwž÷0úß2æXS¿j¶ïMwq­ã3TYL(Xmé(-¶ŽÛºaâ›æ‘Õ=ûÊÑÝ‘ýLxdl[žw—GÔïfHg”J¶Ö~5Py#ϦÐk§«#œg5ο]œ\M¨û/…Öh£šPˆ+Ò¡À@éîÍœ•S¢#ª’‘h­½1|ùÙz'+?ËZ<xT4`v-<Rå” ¬M&ê™Ù1&#4§œç·–ÂðÛ(WnN¦I0å³L¤œ_å|ûülŒ Îx‰«ìGúV÷ŸähHÍ(œûVÍ£,W™&äd0P¹bhPžÊ(†Þ<éU–¾ŽÎnôˆLŽr½";{=LáØC!$È´¥Ã“X‹Zê;ÿ÷‹ÓºçwÇÎ$vpâËÊc#¤Î|ÖT\N>TâÎ:L—¶WnÊšbMrÏa™Y¢¶t‰Šjš$(ÍoûË=çÕ¡Ë¢¨@w†ènÈ°[¤á{Vt9šéóL!°°Eb²7‰V¢ÔÑ \Š5/Ÿ#<j¹0JîçWË$¡ûóó(,EYoUeðÍÍs’[ËD8Y…B,U„Ò‘ËñCKû¡åö¡ìLøvëæû&Ý7:}Ñþ*a÷YÇÉTL á!‹®­6ñC~4aàèá· áÐ.ÇÒæp‰Jm ²¾iéùáÎ3$«æã—q&”ÈLþ)¡/Öª‹q×ʳ2ÆTl8ÝíÂóÞsÖ‡°·—Ý7廘üÞ– â£Ê{D¢¶pŒñ}}ÓcþXyïzËß×. o«nËú²q á KCW ùçö͘^E,QÌm5 Eu6³ë’Œ }9ÚvãÖ5æú÷ótX§ÉÛbA;§Âðà¤yÁ¡÷(1DEç†}Ù”ŸËøk¯§ü8ãg‚G›òì~ÿ»Úèú§è{gñÃÙª2qÅ«-Š°Ù1ª|;o….œøæ1Ý;¬&ÊeŒê[ÕF8EÕ­#<UQÃË'Ö:€®•[ð ò>[;÷ã·!ï˜àÒ#4j¤¾>Þ—ÞC(EÙ÷P'”=¬fÇ\iȉUD|XÁŒ \w·%!Ñê‡]s¦˜oT*¥áªéW‚Ÿ‹)QC\U{œ}zÕÖRßÆ©òçX>µž{ µÛ÷ôÁ… ÝŸjs‰Ôs¦÷^º§«ö²mŠu¡V±F{(–‘‡ƒóøQëÍ;;–›ÕÓºYàTò‡C TÄ¿XnÓ@¶v‘-N¤s΄½oã~wJ,î­/~¶4á±86ÎmQ3ʲ©ä°vÚï$@ç&sµÅ¥èƒšâ#O% e5¦=–Qý*ÑoˆríÍ'–â=QruW¨.5rÚ€äj{LYÙøÅŒíå«Ç6û_•\¦ÆØ7S·•x¦`‰.»›Mwq® sfF#<öVRöÿ7݉æiGk5‚£gôtͽ‰UQ8™ÊÂHjJâ9sëA•@¾Õ‡‚iæœ#4¨Êx4©Éãß[³SÇZ-ŸjzßðƼ»ùlcÇjÕ<SFg2¢*þŽïvîC,ÕR½™·gU¾IÂùŽÃŠmõ‹ý¾5—Ú-ÈtQ§*Qþú(¼ø ÆF>"?ݘcù<88ý½þˆ„|“T¶ß=ñÚƒöÆZù­õ1”~ÐÇ(´:U/áSe¿¿8õ*š¥ph¡œ'ãKóuþÎc:˜67ÇO3ϳtÆ`ÈëRCRŽTX*ÄQnÈbèžÍ¹b°Ìyµ1ëó‚¶OØÀ¬ZèÐ7—#Œ<^ø}ÔýÒ¿WùÑ .ÎipHþÌ$#òGtÓ-icXñÚ¸s,³TX*#<‘RlÂOÀ£y@Q}û»@?_{­Jý¦œÔQ‹ÑƒÉ¥îr±–„Лœ¹Ãÿ?|çÇt[‚#4Ñ)®‘ïzEL‰„(±XÃúÓ1R®¥Ý‹qVë©G·5=™Š¼àYš¨?ƒ]ƒõºvfÏ6u©‚Íêª'0µd+ü»¦D§ P‰–…š¨¬øFʲV‰è›æ‹ wîçë§n+º8˃ڎ6¡vO"ÏŒú#)Ã]L®zÙß.°úîQIáÒÊÏÖ\uŸ\À¯‡r->¦Ç/†W‘p«°lQ3PÖc‰s»¤Å¬+Ilþ:cŒ(‰~ÐáÙP¸®êÀ*ÇÝŽV™QƒI…°?‹xHóøÕQâk#)µED×÷“çH ÔNjDvq¶zµšÊÆ+b>Ë8`i‰jmÑw[£{ç÷þqóªú»Õ)YSW»·7“µÈ¥ÇUÜœQúH°"ªÅO²½3†ØÇ4·ú¾~l" Àëðw7 lå@ýÖ+õ£xv±$âT’K’dùåï«ÚmúŠÚ?·Çû¹°,´}DwÿN!Î+#4góŸµA5('V?[Îó~[~g·´ý<èÎbÕèP@r¨³Ã³íÿwÌ)øã©Ê0SÊ£aòl[|ò-çÇ3ùêN—„:?y°ÊÖˆa|Ð}1cð®i2±¬Pæ*ñ·º¢:øÆT¦6M›|²hA³ì¡œÍf8uŒÁ®[Â4íŒ95jüÛkÅ‹QÆFõH4Í^œí¸¶úblP’s¶y¹òaj¨‰Ñ04Z>Œ’ UWÇÒÉB(q_S"ÁD°QiF!Å2!ä8¥–,âß+œÆì–ûÄ}pó28Å|×£±¢ Å`óðª }ý:£Õ¸Q¡ÖЈ£g¨L˜SëmYÂ.±±qñÑH€òf²4Ш‚7J./bÚâÏ*t½æáÿÜm•J1j¯Ìd^EžÕ.Z¾ï=X<Wš,ï‘™<ÚY-íU#<¶b-U·úe¯úýó;Øà$\»‹hpåݸ¸€ú|`u›½ïÏ4Ötøµ÷³å)ôç9zbƒ$p)wÁ¥Èéê°èš3M«èê"iIU”š!h2oÂP±œjhÞÕ@á^TFÂv4„–^ÿÆ;ÐîíOóáÒ-]ñß’HöY€ßž!ºÛi0Û;ïÙÃ)švˆÇàõ™¦ŠÀúâ2„JGé ßÅ*ïåºm‘†›7¯Ÿ,zû«m#4Sµ]RgóݸB¬™£Ô”ßÇ5y{Þj™`ºxÖ>ªŠiSº¹û+ ²1ü;¹ÎuCþž'b“³ãûC‚èÁñ復Åð®ŽíÎ33…•\óøŠ^žqæ¦=éFdþ•§:#÷Èk.§F{¦ œÏ€‘ðƒsQ´Õ({Ö)q—j6ÓÅ.3´ˆ X0`ÔÆYèL™%•á¹ß=Mz[¯"µ[ÌÜÝœ#Cœ§‘RÁL†$9•>’ $! 8ˆ”DT’¤H`Oq´zVð¾1QfL¸j¹ºr½°û/VzÔºþÇO–ÅíòF{.æV’é(“.)2(u;ç!Þ92× UxäÛ4ß:Œb«˜'‘õ òñWÊGèyíÛƶõé¾ÆéR ïAñ"/R‚ESQÓÛ‡ŸÛñ¨´½Üã½µ¦ÎAõÝánPòJ΃¯ù*vò;ižÚÞÝçÞÁ=sLrÎ!“üJ_›i·¶SèíÿŽ¾Ý3ÕðåF—Çð(S¤Éï Kÿ®ŽžŒ¬©Î9¤ÿ¦(låK¾B2jšmdHrtqðäÙNNlIW»Oø¦§Êº­äÇ[\\ ×pA­oÅ8&µMoÖGg²cÂ#47Ug…ÀoÂ7óQÓXÊŽZ_r Dß8uNfÌ–D³®‹ÿÌŸJò£Œï´¿ï ‚zvN¼óz»gG(û„l°sã2¼_n*Ïí`—zó}¸#<Âv-ÓJ#ãY?CÚ]´Œ' È‹¨ð@\”j,,Qw 2[GH˜×öôè/Çk_,Âèþhé:À”˜yÚí ÝŠCK6»ÕyµÒ>í®{•#4\÷#ñ#8ÐŽªxÅí³±8¿ÅÜß®]c¸åìnzĺeRñFÿñ6ûa.2.s±©¾Z@òÖzÝM§#<¨·‹àmªˆÝÞ3`íéË£ŸÃšë%n°$Åú,Aäë¹SMdç`õM)C¡#4·!½¹Yo"Ë„Rˆ—#)±Ýö=ptçä»—‚†x;ÚºÀ‡B¼^Ü‚ ÃVŒœ—\:H¬îe |å8裡lr’Ï[\âþ#<6ºŠþ¬ä8éõÔï4 7|våõreâDÍ9Q¬Iõj‹zÀó‘åÿÖS–¬9Søßç»ý#/G<ô™Áà„$ª"Ô£ÐÀ_9õfìýwEÇ_¤Æß³»&³×øt›d¡óà:ø <nK…åUþp£#È}ü¬1Ç;‡ÁΈ ³oèPM¾¡&ßòþ›„ˆH’à™ÿ§|Ð$Ÿä¹o{.0yŠíôWY ˜uÜ\’ÉCÒÍË4õaÏ:Hqsâ:ÇxKåëèuYD‹rèaØsø5z¼uT;£Û˜Ø EëþlŽçæUîöY„GÄS¯î]{¢ÃˆvøMÕR4lQ°3·²Æ¿Û5Ï¡Ö ™­i`€D0Ùý½¼=?ºÛÅ㧇ßm°ü4>zq^3mTÇÈ8R±“ôb]¢_{…ªõ¹¬Ù…(£Yi«…·Gþûâh=¹²DdÔÁs¾¨LÄDYæ~Pî&#)ý}K z„o «\û#4 µÓöA³ÓÐ"/]´¸"#ÅßO6ºü~#4ýtÏç‹wZ:£ME¶³\‚½kúyœ¤÷ò¯/MIýÍùMœkLº( š¦Ý͈×SI~yàüö&›• bôªâô1†¤MªñŒŠ%Ož) a™)Ù˜Œym·£Ž¾ÖåóW3úŠo\zDi ÏäöV¥Dß,¿îù€íÿtÆ£B#)ñ˜!#) „ŒgKñ»íl0yÎâæqŒfÍû _/€ÆÕ2ÅeñO¦KüÖ~EõÃWúWð’¹Âà.’EÇËõ=#)¿ã’5½<ðÈAgÛrKžˆöœ¡ùl|1ÒÙÑïo£xúþ?[Æ\n®û¯áê¸Ø¦våábJ܉Èk]óüÔ’>ë'_è}Ã@ÃCè3×Gl²=cOOËÃ+wxß`Ì?(¿ 9>nÈÛðtç˜É#äþ>ÈÆ $$#4­|Y®ü„E¢³åÌ=Ñ÷õuRœŠˆõE#)ª#)táðgöè=Zþ³Ë7)ð¨42!ó·âëïxÎ< XɺQTéî<N\ú~çnfœ8ò*l¾#÷9÷ãøX:“pËé–‚Ê|!y’©«¾ÙJ2ßî^8 SeMÚá¦ûn °‹°©Ò0g7Í®Áƒi–þ÷™“‹‡fíÈ*ìîX€ì ÛökÓ—R¸Ïò?£cœö”pë•TºbWvƃki’EÚÿ'FÌéšÞQUýlÆêMÿy¨å4ÞóñÞ #<P¸_½ÁÄV«xÏ#4¢ñUÞ»Äq#)#<”b‘Wn$íqÙ®n6Ì3èZqù‰9–òÿ."“ L·ŽrJÑ*ÚICx¬Þö$y9´ºxæÍ,e>¸°$‰?7]Eâ,Ë#6ôq.Çøÿ –…ç93O{lŒá'ïïãìt™Diô¢›j¥cœUŒº¸2Ú ¼±†Š®õ˜/‡Î+ûqò!ÀîâCèKNûÑA>´9£$ØmzÅS½4ëìØãj#40Î_¿Æ¼­Ð¯ù-Kr|z54°I»3‹ÄôÅ_ÅŸ¡ÿ9³AØw^„1(6œ÷QÍ°2 ISŸ;é)<±PÏ8“Ü›I±a<Uyuû²p¶ñÈo[ÁÑÂÔ3øw Ê|‘–R2ühï%…¹"„þFüyí†Ã£¤ïí¶w6wk7½eùÿW­Ÿ¿I>æ5×Í.¸êÓõEªàá¼h¼V?²`fÀ⡈y¤¸&øŸ³m+“P#<”‘‡·Bó=Ø~Š¸W/<쇦2f.V7NéÛ#4naçÈÙ>õòä6Ò·9,#)£oÍZf(Û߯¦´ñ_î:õÿ's÷v‘Ûù3Ýaº"ÁÖ„qTETÍ—Ÿ\Y,'yŒf)VÕûÞíRzNša›4TpmuæÁî1˜!Ôgª¶‘\"ôÀ Öd™‚Lã•Ñ¹|8ÞÉ7Êøü°29>ù#4#4®&‰$QFÎú¨ZÇ!®÷X†ÓÅ}¡Æ;?Æ âÛ˜~¢#4ˆp™Ú7üÚ’KéƒÓ*#)-X*&‹‚‘uñáG»×dK’:–Øße IL¬P¨D&Q¬Tiò÷5 ûUÐRY³d@3Œ1-ìxvŽÓ‹+ü–nàõ¿Y”ÒdsÓ¾Å1•ÑÑd”-Ç#)ÝSŽšI¡}¾<¾à/¤#4Y¿ø­DLÍew j%J ½ßÏß Ú£]°¡«£ áNºÇ)¬ÐMP„ °5š’ñ?k…×oÕöá[_c`†ù9b| ¢¤É#4BÀ‹§LÉgõ›ôÚ¤ðU­ºsžrQ BÀèè|šãFm=AbÐG#4ÒªˆÀÇ#4,U²ÇÙë6e‹¹!HbC(¡S‘¥08ÆÐæ„»”ÆB¢e…—w9cƒk"+|ë¦m|½Ÿ[×Ãäç{×$EÊ㦠ƃ>þù¾Vy[õkUtGÚ]WìÌ ööCÕëáùë qyJŒeE` ÄUô/7¯*·‡y©1º5^;qò1·ÇqÃñþåÌÙ¡5pÜf`—ØHH ÏΦ@Ž¯Ã|ÓÉ]¿«¨h>ê¢[”e·œ7eu¬#)ëòÂPWâwwêÊàÆ#)l«ÛS+m˜‚êÀ@†&Ãéý˜0ÄeÎÙ¯¶ûþ¥©Áô‡¤^Ï®Ãj¾ðt¢”ž…Ü™±ž±”Jh>îø:áEС`ƒ÷­ªÑHÓ©é¦UW‡Þ©F!kj„|;kÑ;iU"àÎÇ0)1- #)2#<6c=$Ç7jœÜÁšËÚ]¼_ˆååÄ—/æLJn#4Í÷Ïh‚¤táGJvè±ï²ô¸oWá!rÑåÖZ¬sazù?gß•ÀM¸­˜Ì0´ õØe5EÐYîÎ,šs¿MnB:,W,(ŒÏ*Ö‘“9²z“¡êUlÁü1åÌX蛎»št»r3­Ÿ®Ž†9·Ã–ˆyøO^·Áv^5<"¸7;­‹ßBáçÁJèІ¸99ptbÔd*GN©.ï«)c9pÏC¢9Í뙺ÞXv}Cvfú}[±N¹ù¹j]¹P3oàáƱŠé±¯H\¥W¥ÏF8hŠR8¥c—aÑù·ccÌÁÖKÜßÍ£œIºL%® ;>¸òëðÑä½0ÒÛ¥Ý0ezò:ÝTjyÅYdÛêÊ,ž|Ôé¨ÚêúzäÎÛŠ%/)Ì~ÝÍZãúkØÊYEŒùÙ-Tvùq<S8-y°®™ö)êv©×Þ6Û üñs§v¬…Å´8o‚˜Î=„#<¸s²#<r¤ŸRŽ‚ì{˜kÆbÂØâî‹ =µ¬Ý²]Pˆ†%ó»]Þô»\ZyÙŒh¹ºEÃkšÅPÄɵEñ̘(•ÚƒÃ><µgðÜ/^H<ìŽ w)´ÂBÍD6 ±°C,m#4S˜ˆz ,Är‰í×ÃÄÑ0iP¸”ˆ1¸ã @6Â;È™ Ã{†¢ŠEŸ_§žù“ƒŒG‹†XÇqtÑ©;¤ÆðØH¯tiX²'´¢ÓÊ?„ì©Ë#<èøãPY?#<õ†·*Éå.S4#4À¥“cû4’Ô.´®L;;Û}Îâ¹ÏI8ã'#4Âõš¢CõR–]Ž ÝÍ9&ÎÀ$¦E ß8Uã&B®¨)à†àwQNFLÆT`€À­´<n±§gMa¢Kn9‚ÃSé ™‹N‡µÓ×Â+ b!ÈmóêMòìðÇsJ7øI½ªs½mÒ8üñãéZ-lì.7|‘Êñ¢” …°Æ±A' ?EÛ `,\:Ÿiɳ“Ž¬ÚV –åà³ác±¨"$ÿÞý:¼šQ9#4žÑcZ/‡iݘöHÆKVpNðÁœ-#4Ú˜¦Ðé˜JÞ)”+˜+FÚÝAðÔ×a‹i3† j!¨††¡ÚÓ"Q…iF!†U£Cb'ôB f34V—,XŒdFLÙBë,y#*+­Q Áˆ3»>™ü;ƒÍ×èžš’¾Ýo^\9Ž*ál‚†È0X #„ºjøÞîÕ~E|ûuW ‹cLŒQ`š2Ñ·ÒK_fÞhÙ´i ²%à f1[<÷#<É› c“—µ¯-ynW˜µðsR›Y &a!±»¶§£ãÜÝ>˜8ÌîFþ~þ-ÕyzLÈQ¾Ñ ¼§}»™YÌÁ®«Ñ}³~{:Æð>+}FÛ!•.ã[Ü“¼u{)È­Æm•—¦Â‚è³–‰#DÇ v ã´g6Dáà2íÝåOå i³+;œ›¬§³ÝX‰ÑÃJq>Ø׈ÄBQPo/#4Çäþ>`+Ñ`“¯Géd³8Rú £Ðÿ?UErdy^Bo{GZ†)Ûº6”ó'T„ÎÛ?IŒQ€ò,Ë‚ã•Ðæó¡y3{¸›w»G»§w·Æ7©ƒ_NÃÒç—§ì©rÚÑ8r&Aö‡l—f4:Y@“B`jGÖZÍÌGKÇj)°¨ûlÎ1ð»’µgŽSâ7}‹fý‹P[a ŸùÍ>ÎFx^<¥^O«ã9Ãß”„ñ;‡ý¬:~‹ß èc¸½‡#<eæœ0º)[”kÅc7C&Þ$òid LþùAÇ•ñ¶õýS>.11°æ?[xís³¶8ïNüèþ›ø³¿>†³>)&Ý<f¼ã}®é<êZÛoSÎì*pã[¡t}œ+L¼†·$Ö©(‚Š‘(00²Z#<Éá8‘¶|T—nÆ8ýs²éƒ#<Ûøß¹0YEà‡†+Wz+V\¡ˆøÜ’ZÓØ– …#)Ä~¾ §Ï“òZ©ªßý­ ²Ð÷Zqð$. 8'uJäÓ¤ÿyÛ%¤mï†)¥Ê¿Hàû>æ4ù¢.‡Ã“`Àrhm1M¤=YâL²"²#46réLÔÄE…Xa)&‘Aµ’Pȳ"$éÖ‹i'ÖG\Å4Ô)c¤QG¼m mA&ñ =7üBà&¤i¥ÖZã!¢„Ö¨o.Ø%v¢ˆ¡£JÅУ ·QÌš"Ø™#<È–œC­¸:ö1Äa)•L¨aÊbfU*E $e’ªðI/<V7»M3òx%¥&÷“úÆTS_ÆzI];Q<ys ¯—8\~|óçÏ:ŠcI¼‹‚†•Â>þÐóÔJX¿=(,ø5áz>u8—X_ið³áéT:肸¼ëWÁŽãÚŸ=Ó‘å ŸØåH·CY‹\%õ=A{ÙýÁ¶QèZ>ë•ùAT.‡T@¬n8ÆK-Ó¹ÎØhØ– eb»/žò,¢«{gí®}e˜ã7nØ´C&Ï£û‘]z(Ûó9h#4Ê\† 'n×,Ío‹Ÿ#çÞ/t‚Þæiv6Âs*3*6Ž^%eƒýø#l+2Zd}^ÕeŒ¨w|zGjˆ”×pŸ(uMM®C½”l¾éSä0ÛeF׶ɼMRÌüÚ4<5Ñù#)ú 0Z8¨pY–r*£e‡rGÊDÊ‘&{*9Få :$éÂã©}‘ꤢc¸ 4ó$„Á;>)Í™]¸‹,.%’?ßœ)Ç~ööÔ>ƒÞk#e˜”Âû sh0Â*1ÑΔ¯È–5îíéÏñsˆÐZ¬ñssFlªuÕªž•Ý›ß<EíS©"â‰4Ðî1‰xøâ>«Ëo~_†`côðÆíÙÕ㉑F243étimË“*ÒDbŠš¨bŠ½zÕ»k.iý¿2o©“hÊE=N¬ªh’¦:àœ=Ûï´ìß|ç–O™+¾[>†Ê>T÷bŒ{qWó×Ç@©´ÿÛ´D4…ÖRôB™¦vï·ûº¯àcë‘UŒóû­~¾ ÊNô÷àû#X¨MÃH¦Ç,È€R7ouúÕäœÛ‚q@Žk,Çd؈@zpÿt(kÀØaU´üâ´ld¶Œ7vÆ­tØ\×$֊ܶÝ5f–î³ð}Qb!c¬]¥²V#0[ì;z#4D‰„J-k?zº5ÛvÚ°·#)B9CŠûIÖ3©3„o©è÷üG³\±pÔ~ïhRFãÜОlâ‚}à‰J ·ÑáÌ¿ßW2AqðÐ(r¬Óñ>Çï‡#4é¥è¶`u•†¯ªm7Fïsç/k#)ÓÏu z»@ÍC†°:okª:fu°5gèÃðÜ?ð¹ ?;ökêíM$$¹(@F%¼éäˆï*Ø=ÒÁ™~åæèåŽ~ü†nä-Ë@ÄSàÄ#4!Öˆë_Ô¾Ê9\¤šÿIvºùŧé¡ÄE>™9¿gÚGeŸ÷¢R—ÙÛQ{÷­°ßÊü`,| É3ÆXÒ÷öı9ÏdÖäÆp£§EŽ#4ÿ-ÖÕ’‘=òßÚÿ7cêxùö‚4÷.‚.Ó: X÷Asðûÿl¹û>ž _6€9T Q¬Í̬ŒS²žñÂ\¾ÝKM››Õ;úº-¥Çé³ ØœŠ‘n7S¯^‚€xù¹}xm–ßI•¶¡<°Ò>l\=>ïC¸Ôþï/鯺ûgɘº-ëòC!ëÕ·x˜âyÙ€ñ>†lx™‹Š€èÉWžoB z¥ÀtYÁ¸Œ´ÝL„¥ª~'$Á.ÈgvVru}Z„E3µs@@üÎ ´¸î$‘eÂƊܶ´Q§j­¢¾}º=Çúº0ã’T1d€{¿gñô×èø™f&kÞ<ã8•W”ûûÁ=}Ù}¸ø*ø—·&dt“ɨڡ §@§Kšf=œ§ïþÏ/ð¤â‡Ñÿ­ Aîz|l°òý;»ž ³Õõ}Ÿ75½€ º—âÈUP2ª %4uùµß„,ÕNç#4#)”|¥G«2yéÍÔ"s’60ÒoÞßÓõ·Íþ-ó¶"¡E±òvÙýî·¦mIdÑ3²¥Œ¢þ‡‰£MŠ¿Ñ_GmÚÞë{κºjæ¼%?G‘,(kû@ú…[“´[Wìü…à~`D&’¶)4¡.=•Ç[0’8ÈöãDêc€ÓbƒMs›ü¯~o¦5y \«ºín¤Œi¤Š¤óm´^©¯?Ó_‡ù¦¼ÿŸcèýòa àtX¡EäøÔGî‘V$1ƒgåþöúk°üjcIú*ÑÑQ—–‘‰ºO¹) ¨"À?Î) J©‡‹à {Ò“õÅrE¹Ñ¢$#­#<†Ø ÝýÔ@ Úù*âo‰#Vüªô ·*Š¢ˆ¡åŸ£þŸ»éUy¬$7Ø®Cš·f¤ii“UÿlšpüÜAÓH,0ʨTž^em/ýxí4Ñý{zŽm“X8/ W2hoâä¾–‡àÊôé(YöЩìE`ÁÑÉ÷ïÊ“roœÉ "’îEÆr×TrCýÐò6ÇÀkWoÒ¨:Ä|¥¥%‹Ð¾c2Ê磓ÓйLäu£Qk/ËçŽ8ôñÆ6þk8\åa‘²žÔ‡Ä2›¤òk²Ähð1±`Ùì“þjZá¡‚°~t冭·Ãî®o$ù#âû¥B†#<0Àþû“N5S"‹Å#<Òl&IÝÈ,U)j—Y í«þðÕ3®wKúT†li;þi.l:<Qéˆþ–QØÒE »¥„…Cû?;Q –(«dþt¤“a›ö^î„?u”¾8L¿–}¼³mŸÍÐy¾k;~×~ø;D¾£ézúÉÅ'™õh~~²BÞ ·Þ½d«‡ì[ËíߦÆzÔÿÔ»=?‰L?[Õ,ˆÛòwÂ"Vì5Ál9ó…ØZèGå²?›‚¼>Ü»ì`[oF²š-£tHkîý±pÄ#)œ“$UÙ‹¹—mâþ»-¥9Œ¤9ÌïyR¾:oš^öü £ïŽî¾6›Ô,k¿ä_ŒSûé¸ñýâËÈu¹g{¿Ã‡’<#ñ6ÝU•Ïýú‡oðsõeÜ®JêaËùéW±‘7•Rz§V¯%¥@#<Q?#`Çuéî½KÞ¡ÚpjßuÖÑnv5·ö8aËA¸sýJ“2q…¬ºGÙê†rš$JK ÛôÙX»Kœî}T¿²þ5Eœ[´éጾ9y4lõP(õrt_<îU5]¯nWyJ6 ¤ *£ÎÁÂá`Ëa㌺¢îžu‰KHúˆþ6,ŽÂ>ÁÞ=a~µ@£áSðè€àø wô„¤ÿÌž†ÅåMhr,0§ØÃÂÿ×z×àùEF«e0@#<”˜‘°úŸÒ"<5f‡@Þ1¬RY}øáßÓ³Ò¤ØÌÀÛ³,U£›õŠ&Úv?ÕäžîúGÂ,Æ_39 åú¤|¸ËOˆ"%I†aå×åæk‹åc“Œ?s#kQêM™Õ²Î®Z/!@W«'J•Ñ!ƒ~Ÿ `Gôx çºÀç(´6PÄ#4#'$y–tÎaËäÌQ¬!`TP˜íegx¨ÃWé`û°úÉQ„•Éa…~B‘F±Â,åã$DF¨ "…àý#<©ë]ˆiçþž÷HŠLFPCx`–vû€öûÓ*ýÅSõQµÒ„õ‰‹*ÌñúgG)«-5aC%Db#4E•«5xP¼Š¢H¯€J¯Ï¹žF*:œñ3ÆË €ÄŽ+ú´0¶j ˆbWÖ¬I®˜†ƒ‚Ä…ð­j¾qbOO'ÏñíUÑI<YHRÝT'¹v‹²´´÷ky§»?E»?½Ý´ñížkáíãÏùö/ðªÆ»ƒOê5ƒoìåÔ‡M…¯²XÂ’Ò2S ZÀBYäžR0žI#<‚ŠÀ¶MÆÿnßÀÖ-Œˆ?Ú}ëLj–l¦7`þÄØ«ýb/vþÜzwþ]ž?OÛæ移íû\~fîžè* øý Óí¦Z’›>¯ºÛlääõ~«=®Î¬ëµ×>©%-Zè¿:.Zììwµ‘l¿¿•ÝþÓøpÝ…¯Îí=›>îßÔ4'#4ÿ9¹.½åzaèÇ<¢¢[*öþ_ÕøtvµK‡×æ_O‚Ü:þŽzÿ—Ó¢—Ôïpbtp×ð`ä<Ëø¥7ùˆ´y»æÃŒ¿.aïÛåý££XÕë-€úóäÓ³ö~Þï±çŽÆ ¬Z—£'»céÝ:ËVΟV·#44´{¾ëºß£pϤzäÃÝÛ$¦ô÷ÿ9|±æG!#)Ë4€Ø|¿w¥øÅsðØöéÇ£L§Èb-Ãk¹GÜ©ËŸÏæþÖþJZ‚›h;^ËÜŒ=X\Û¯¶³äÃ¥xáۮɷÊ7[áÂéÇ™Èîë)³Ý#<1œ-³È?Dý4pt{¾MŸykOFœS¦}ÚvÆ0ÓVÕ½ã«\Ïmy÷n:¡}ϘֱêC¹|°Ì]ˆ[ÈåG:^ĵ»¬òNûL8b¶v´v9Ñ"ÖÕÑûõü­€7YfÙ[fØ›l½€ÝϰƱÙɲ˜—¢Þ,>ØØ9;¾}Þk{ÔMS<qÚ-ÛÊ!0X¹›H]«—½ÏW4ýŸœFÌlÓ6É;—ÙÏËŽû|ÜyðóïáðÛ‰ºXïw¤‰|>î®÷Îýwáä¶<8¶{öÜÐó(aÎbǖɺî=#<Ò8ÓßrðømžÇôëu…Hw£¨ž´+ŠX{¹”XUÃ_ûÜów§c 9•–5ÙÑ>ÇòG»“`ÁÙGR»ö¼pûBkú== ÛûLþ/?”{““žËé8ŽóCðoØ4[!g†ù_ðƧ}zò#åÁg%½¡ûŸO϶»²·¯kïÃhºÏ§üª¯ö¯ J»0‹½ypdû'x`W4¼*ðɱä]½ú>± ÙúŽ*+H#4oF6é³ÏöMâý-³—N2ä‡5%ÊÞáØ#¢Ÿ«Oê®>®0÷œ3µGWóQB>š°3NÍ+÷}#4ú7thúñ]Ú¿'q8×™?7·úý%MW~}»ËµèÆ1ø¢°Z?¹¸:¾ï%þÙó$¿ô˜íùŸ!¯2öóŽ_£§òÝœ(=6ŽÿßN”X.b»‚Žì¼>»;²µ2ü›“®fîW}6ÆíÙÇ8õ3¦?SÔFÆŽ…žÿ¥¾ƒŒ£€ó]ô7ùvÆîúšÿ^z°<aM.z90è`GŸÀvâñuý^iÛ³›ßþ]w²ÛOG?>ǧÚ~ãôŸ³î?yoàuøjßÉn<ÑM½ïÓæñ­lýÖ’£W¬Ãöþ`Š¯§îéÌgå—¡²ivúúþÍ_K«ü»P{Õ#<¢   b¨Z#<#&*ìõ|~ŸatÿòóÉYë¬îìEÊëçU£ýl®¾Øj#<ÏìKj’ëçùÚ%ð·ç 1û®QpH#<./E€wÓQ—n·ô?ðoHÜUŠä“ñýÿhOÇÄu[êîú÷~ŸšþŒyõ³äÏNÙù<M?p¹™¶xöì3õOcŸ°ïüî¿=¡áE,¯NÍš—fß›^þsRf”N_Á~¤ø|x»‡PìñïÆ#òèèÄ|~Jxr÷~<Œ<3ÿ?½Ý#<>1÷Ž¡Þ< 啾HyGî#<=Oi•ÇèžÊf¼ÃÆäüì}oöýà 9VZ¬~ÅDp?’ýØa‡æoåÁ.—´of)án½Ýãïï×»U³¿wÍ]n“‡3Vïáò_Ù`Ü,þ$èº<Ÿ§9n‚c#4ºçâ·Ž˜ý¾£Ì~:“Ä~vœs õÊAãçÄ” øˆìöusñçã/æ>Ôµ<“«”ssÁÊ­òÈ0ú‹5<±xà§ó ÚÏÝÊÔø!P„-ˆžO^MÙ„ÜŽ¢%BExoðN<¸s<i¿YÛú{óï^##4´[×›š¦G£ò°ôJŸ1b¿{r»0|Øws.®w òVdÉ}ÇÂvÓË4Qó‚<§ã„,4Ž#ÍË#<yÕF:pñ˜¾ï¯`7_NN¿“™%®$60IðË=“Ó0tZ¬cÀ¨Nf 0Ï¢ûT(²Ý][Eèd–“2A7:%lÀ #)£Öj¼Cš®ñ`€d>”"r:J핌렦¢i^Œy4βÆWt9~‹Ýžt¯MüüŽ¼r¥CªDG[0Ÿ]±ÀAëò(—gÓƉt*™òÑÄYÁx;.Ù™À’Xê%ùo‡(é6!¡7þÎ:ìôÏWÏo´'¢ø{x~±‡ÅÝXqi4ûŸÐá#)[’×O^7A¯h2òºcñuœ“­•ª+‡?*ÉÆ}ñuï°TŠt>|«#5hT2ÑTFmA:¬`ÏpÍi3bÏ;–îZåf­clT¼Ô¡©õ§µæï=—}k¯iƒ­‡@ž|à †þƒúGDÌ^DvC³£Sd¶[Ʊ.¨èß\C«h¢ #ëçm€+¤ÛÉ?§Øñ«=É*Û¦9Ë ‚æR­ .lƒkЃ1£P¿x¦VÐh•ØhŸ¤UÃO£ñxÕñŠ<}'fKé–÷cƒîùÓ@¨’IIÜE#<ŽÍoˆxï;NÐ4꜋¤ºLBÖöo5îùäÛ#4KOž‹.1óÐ@jâ>OÇ„W‡»÷ÛoÓÉbûÇ5áßïÀXtO°ø\´#Æ6-‘^#4ä²ú˜X1Ì1#)NÄ7$z‚§Ù#<:Ž“=GI.SQEG˜[Ö“‡ÑN÷BC‘Gv­@VýviéÑg–­ßÌwç{ñ¿û¿8øC¯Ó÷†e§Èí3áQ³o$)Rܧ§Y3f~Eñ•±àzö"BO~?×Ê~“/'À78¦OÑn#žð5ßì“laé<®óèÚ»\hò¿Ãµ¯‡Î£èÉ4ݦýÞ«ý1‰ê©LRI͵ôÊ]òuŒ¤öÆ/ulu³ÜYÌrs~züá'wþYÃW¦£Mÿ#)Ðæò#)ÔÔƒ’«Õ&= I(×@ѸþÅGõþL•Jsè‘äw«ø‡ÔEì“–U—›éõ¯Á¥€öóËf#4A°¾:ÀÉG~ß—Ç·û±Úi°±~ø±Xÿád?ÔZŒÌ¸ÓÈÚ+nFåe„‚i¸ÁFA„aIåí€V6j”A¶ÅHâæ#<+EJ$Ó¶ #D49ŒŒcIàâUV”nmM¶‡÷nÔò$1¶¸}=3†´Ö¤Olƒ&D­#u„ c±GMQ5†5i’SjÅCCt!Ij²4±[a™F6ÛŽF#@âŠìõdÓ«ÃÂ,ŠC„dÀ¤ªˆíú´®õ]]·K¡.F¶B·QhJýñÁȲm:ÊãOëþ[ákGŒˆ®pã9#Jþ;ÿ«zŠ2]¸n¬,lÄ|ÀZ¨9-ì˜eþíkžãDîBÂàÝ¢š½6¦ ^%ï_N]›Ãóߦ>ßÓLôø<<‚/F{ýHmi[S Hèê¨)3ŒNY&bæP²`¹,KJŒDL!jÔ&ØâdlÂÒITõ" 9(%]&nøNÏ»bhœnú4|þ\®YÚ¢½?"iôSõcÝ.³† ýzò¥ðì"+¡âᮦÛuË( Îw+V»\Û¬óûÞ:<Õ×æêù*}sûŸ»wããìà%ê*õù±Ò¿£îø#Þ§?dz³«‡7Ü<°Na<ÓÉû¿¦·;>¬?vR?—®AÛxˆëÈXÙr]†Í=K(Jí•_ëx‹©Uƒ_ ÛeÖáó×lt•¿íÑÑ“»9|b¢ñ´.¡ã#)á §™Ãf~TÐIAÛƒ¼¥Úìhõø|»ð_·—âÕ¯ÝâQ-B<åxŽž.4ª›}¶^ÏjðígóÊî<>!Ó’óÇgµy´ ÂÂwþ7»ö¹\ç3Ü®µÍ0ÃÀý¾ÙŒz?®Ý#<›”Ëòíòþ­?-6ù›wô»o6Éón(›î·÷Ï~žù‹d²a|_h[ûý vzUE#îAÒ+îÔ.8Ùñ8aþw;cF´×™ŠÜÜAL1b¥I¸Äÿš¤xÆ)Nù/³¶ 15¥#4ÒÁ¬!X6b¶‘¶îo{'œÖe*M‘D—iÕK1¶’;0 u¬*ðÔš†ʪ(׺‘³yk4FÀd€È£°®ÈXF†GUÈ”¬¶v¨´`ŠhyÊ«W9¨c|’Þn/f0¶Pˆ; T+) 1Á¶#<2Anäã8¹Ù˜R1¯„©H#üøhW2p‰¹¡ipb šAZ0<­êivÙ.›âpcª$Ú1¤«L*&)Å©Övª-!ñi«Œ®!ñI‚Õ-®kD´8ož) ¢#4­”i¯ñá Ù¥ cΣhŒ7³Yhlï‚m&PÍ»‚Üuºs*/;…(¹”TÆtY¥@¯‰ÉGUh"ƒ<ª# Ýh«Å2£€Ù]d¤cÙ=ÁB2tº³@gò>Ua‰ ¤’ATðšP‡‚»YC¥el‘ÙÕ/_l‡†Þ¬wC÷|G'@PV»?@ѶYþöåè…ÂWã‰q¸¬ Õ¸\ø7œ(‹@«%úZ/µ;Ët¾…E'§]‹§»?§ôXžØüK³³I»×¢š"Z0d•O)èÿÏ#4H¶¸|pú/WggÊ=»:¶Å#4þ2ßÝ­¨ö)ÿoŠòLéÑŽe¾Óû5?~»»sÃÙã;cؾåù‹õKŽ˜ÈÛ!!JóyW®{)ņ[“Ø»m4ìÙyØ~§«áî#o'øµƒ*ÉØo”ƒµºÖÚõþ+•¶6øTþkáت/ˆ^óx!w‘ÔH.FWôî ‚#)Çq郆dàí«í >ÞG²‘ÓÞ‚cO¢>£’ z2èºaE©ÅËîõ2ðP{‰$¹ÊªærèÁ‡\]>³H˜œîìõ1ò#4N·ïßk½zf_ëžÉÍ?OøÎC©ôcú Ú~‡¼H?«òmñÇJç;œîËl愶øŸwn·“î«öÄõª¨Âƒ•Oúvt³«ô®\ã+¹~¹‚^~>§»mš‰Øhd! Á­ýùþˆéPÑiTR#4DÔmYwtÍ’XÓÉxr<_hÅ2ÑS[ÜòIƒH^H˜wñ{vv#4"ÿ“¼ÔuË®§çÃO/I™âeZ{½Úoý_ªŠì’5@?)# TXBRŠV›,mªFÆHG’"Ä5þ¶+`ü™tÞm¢^ŠFA»²²ª#"Ùg«Õ/íÎGËæ³ãì_•¤oÄc>îïšwñV'ª·±"€‰,)©ëÛ‰’"¹Žeõ`ǶŠuô Ü12-7Èš© Àt`Ú)™X¸#4MT(Ù¨Ä<2Æaž¤1ãš¹>4U†2j 6äŒÁààŽ Ì2FÊ£&a‚ÌF@*ù½\Ú^%ïÀˆÄÓ#<÷mºo˜ò8´èë§o·#<ш;ü¹A³`ØÙ7cUPßÙ†hjÁFâ~ S,ÄT®íÎ#4UlZ!B8vx ÌŽT#š´>5Q\É·í Ö]WYEQ”¬p²(DάL£Rb‰±˜VŽV†i¦Ñˆ0Û(öÇâÁ£bÑà»»4§PÝ‘EÂcm'[C®Ù`Ø2éº%ÈeÌ»a¦(ã´ÿü½4.ßgXoÚº„m´ÛF9ƒ#l„ƒlóv™FۖĪƒm‰=² *ÛÍL<·+gxM Œf`xçƒò0´ d BÃ÷Žùg *R€…*#4½7mãÅñÛEáuüOÏú\xôT~ãýjã‡GŸë£Cˆ’„)Y6Ayjšœïk÷ÙøN&aŒýæˆ=@¨W3¿‡G<Ðt‚3à?7ò›±Ó7I=\¼eþ`'Ö~Q,»z½=wØû=„÷‘÷Þ¯U¥íz¸Á¼¼%âT,²Û1bÂÒª§ì¥U0iÆ‘ú.ðÇ»~¾mHôEÍÆNQ":~SU<‹#©Å HÈ‘#eL!¼Á߲ƫzA:Šƒ}³F3ÚrŠûÜe÷‡CX8XwÚi4³sÌK&KACª‚;B+£oF°4€ÒZ%méhƒDF<ê0uQíѱ¼Ú°œî®©UZ*„B§-¼=ÿwá¡Û¶÷moÃ8Pò–¨´¼9Ë{]q †èÕ¿üGòyýÙß³×îçþ<{:¢<}9dï^¡~zûÆiòä=ÓÉvÑ y“QºC‘X?¡EG‡átq&‚EËJÐdn-ЉŽŽÒƒë1Úá2æX&ÈREAWîª05 LO¢F:»S 0dh}äu·}ª£);ºŽËˆQ°ÄÜpî;sä⿲ù^=Ïžš{^¹ºÈ®Ç#T½‘Þ8•‘Cg\ÌÎñqE³‚›˜J©²DaDqaBM¦QÀ0Fµ 3~0Ä,B7Õ2•(¦%éPê¨i–ò ôd1sd©£„MpKGj£$;#4ʺs|6­l]é7w.²I‘QÒCÛVh„àÂZi#)Ôœna‘6bés“*:­HT+•ñ`ffŒ´…² GÏ­FjOå„+kÌ}Ì‚é8¸Ü#<ØݱX„ (e’,¶Y#FS!(HöcÊïwÖÒ-Ú•ød©¢¸f"d2ô0ûñÒ]ù¹b(Ç­Ìo-þQ½^s™¬3 ù3ÂßLnÌ> ež(aFòÏOFh&RcúÕ@&â6轟¶ýqm–YÄ+2ʬL£Äób°sc;èòh|™’ΨYßIcú,}–.ßµu°øuiÃF”‹ÄçsYÙz•¤äžIb’tFD7Z6žY›#Ô‹íw¸¹šjºõßðfÛ{p„'œŽRãåÖøfFÜ›’+[]RÜc¬mV±–Ò-àLH›J¨„îû™ðZð€ÖÄCìókà¼[wÑ{ék­O\0eX&Ê(‘ظyL“¨NM&mˆS%&ټŠ±AAO•`PwÍÂjŒßò¥WÁå í'l‰£BmXZúM­ËŒ ‘•Àtx@ÃæQülfœÉmBfm`wI¶v8-Ì-µÐ¡Æ9›âmpלÙôÏl0¨ß:´Áþˆ*¬8qŒJÉ}oèm«$o€¾/oJp$³«Oµ'º3zÖ#4Æpß3,\7¹Áú&,áÅ·ê4°ilÔÚ)Û üÆÂ#42¸mËêjHpÜ LͤÌx[wÅ/»®v¼îÑŽ±­o¢ZòŒ4 &ÂﯥÔ6[›ðøÓ›õºÈïj¡$’Ë5$-ßsœ³Ó³9‘ò<rÙð»<(í–8\ø¬ëÍŸ©†ÞoÄ;›@G#-Ô Tm)E¼[ç º¶µÚ¹Q”—VŒÈkq¥¸".óá2X˜êÎh9)fãæӽ»Š6ڎî‰È+&óÉXîÁiÎ6#<9¼¶wM[á©tÅÆB¬'áA¬Xlm­ow;ˆ÷®‘Z±Ú:1M×W\g3É›*gÈqioÌ\ ð¥¡Ó[±Æ­Î¸êìÎͧŠNt/96£7ÙÉ3ÎB(K¦f´WgßÃʤêɹ˛îdäO$ÚÔša´ó6f »ì÷¬lLxt¬[ô½™¶m¸ƒíüå,GÍd]<]Ðë<{gù¼ä¯7b“(Ó®ªŠ—ˆxU!I=æ<6=¾;eµ9`sgÕÑÐNL½ÑÙŒaÆ%äÉwùÝ*j¶ÏCI_û7q98Ø¢`ßh•ÎÝ“õ6SxëŽ'7#<ª}Ñ9wþz DÖ9n1‘ND\Ì;íO‹š\ºˆµŒDäÖê0¶SqÐC£4dÄ#4˜ÈeMÌ:ïŽñ¥¬¬¥‘cy§¹"h³»[D¦š8Ì!i̪>;Ü˾iá@Ò‘^ù¶§(­Õ ½¨ò}B`„„É0&ïÍÑEŸóú[tèô®œvðÎ#4ã1BiÆùFõþãºp¾ß¾Š·Nîc¾k#<øñzí>£¢#451c‹½Àfòø†1ÔÇeÚtuZ[NÁ ðž`5Bi¹g%l¹Î2Æ%$õ¶Mz3¼¢!ôûÌM*KNó:Á·ˆ„¢-J9!¾ç3ÓLDE`³VlÙ%t_u vØ «„”o“tÒµEã Û ô°¼-Ò¤$ÞŠch4XöNYËWE¯DÛ¥y ÙÛ6Õ{çõ]¾œ89ÀÜrüêû%Ðnì<§Ñ®uÚåÙìtI¹x]E2£Ql¶iÒ`±7ç¡1žÎX!†¦ûÒn›4Ø”…°¦*”qŒKiš8ºŒF#<êÓL¥vû_TxuŸDn¬Fßšñè%†oŒ—8“Ög†ïÞ]™&-x4IyXàV´Ù—nºáo„Ñëž½ #<ðP<`ϳ#)/j#<:¸½Ù¿ÄÀF¸Ä‹_¯fàj(7§iV®ßËã…(tŠA#ÀvEÈ>8ÈÖ—½•úx˜ÐÝιcÃÙóàW9@¶3im/Óñ¯pòé`ºuu÷¹rÉÁ‚ëO8wÔ/Ÿîš~@ÇS”(×OØhlâõrÒWaoäJ°Ü<9e =†¶ãpÿ r IΚ^)>¯#¹D€¿˜rz~ÜeeÕ`DFz‡í/‚ ~„¾ƒ¸ÿW#<^£¿7ž/ª¸| ›f|§Ç{´/σ-Úà.twúo‡˜ã;l¢ò’i<3€"~›Q wuI$è/jº¤½C{fb˜w(—‚Èë2†™´Ò,ô3 QèKŠˆêèæñaJ¬°ïËù/ˆ®9ø {í±€B‹µbáÚ ,à#<y)†ŸŸöÕº¸%·b§³@Þ?#<(·<‡·=#)éŽKXý8p_=ý½óÇ9¯ÇptjÜÙg³:OÁ/Ñ1IÏñ¹ž¼†|:œÜý#4Õq5Q Ì4Ø?pÁraK؉ ziè+xrßâ)? ïûÈá½Ì›Î½#<+%êwcì^×c§ÐêçñlÀêoÒG¸ŸŽZq+;NLÔÔJnÀíŒMÛ`d*JÔ‹øc*ýïÓû%±™ØT9Ë»ý7#4' -æhß½., E(—€Âû&LÅSå1ºùeÃxç²)^ûˆ~-'4æ$(AFó«âKtnF0ÉŒ0õ$W£c¹“¶›û?ªs ÉÂ$\¸J-õ5MSD{ÅðíÛ™Øó}iöܸid'x=›ÏŒ˜šu…'ÄKÁœbæ#)} ¶8ÎØKQdBLIªÙNæµEªÇÁyjáݯ=d¤!Róˆ"Ù™l‰t õlp)ðëZ—Âa\øœ8,fÿ­ÿÿ5¾àæýöýžTGusÛU±tù‡ÃLú2;¯À£Ü+oTußÏiÖxT±©§Ê2‹6^ˆ4Mß4CÛzã«A1رªðoM¦ªª—¢Û­ì†¸É»ËövŸ_G³ç®# ˆDs˜ Êd»»[yd -;.N©¨“‘ìGá;ɯ@zd¤žº¯DlQÝœX[vLGWaГ0᜚7×¥7®½'n¾›§Z¹5Y=%ÃúÊÔl…ÓÔ.¬Xñ¸]g˜ð×'Úv¯âÈÍí‘äPÓ™§—ff¤¤m}Dôl„åý’ø®7¥#4Ÿá{ö謯Juiê<4Ÿ¥¶¹=ìy³•ëÑ›7—_‹Z—I2ÛN€ÇÖoæ—·>LfºÎ~B†Cní}Ji“ˆ+ÄØètòç;kr¹;ÝËñd7þ†³fÅ#4áÁbÁ¬."ë‚Ê'ô®T‚zä75ÄËvb8s–M?-´´˜‘Ûl()4£ 5ÅAÕ…é¬9óÀðÍHZaÑhJ÷êJÐxœG9øe½ûeºôØè±±nY,Pʹ®#4U±Ÿõ"Î[#4?V—}ayâ3 ‚1Àæ Ñ\÷¡†CVÄÃÝ`²HAÐE¼Þ‡0¶:XO”‚BØmŒK=èµÜñæÃ-ÁÀÛtåµBä`Âéj =ð/†|jààø··ŽÆèà÷¡Bq*‡D'$”(SÓ­·¿•ç±/˜æR†1Ê2Iþævé»OuÕ¡ÌìË»LƒÉQÁ×áE WïC{§oyÇSʼvf#4 =ÑÞ-Ç/¤Æu)@9¢æ¥+Z’©nâ¦ø•‚`qÉP Jæ@ó+ÏñaÍpn õ†s=”^æxÂ.[7e EÝE”µáà~b5åg³5b¿˜ã}ÙIÓ£»¦#4®PtUO6 Šý±x,¯7‘q_Lævƒ#)e*ˆdƒÛg¤CÖæ\>æO¶_k ÞßèÜw6ìî«ÃÎg^U¯>}öÐ$q[4up¥ÔMíÖ„[´÷¨& j鑦¿…t‹Îw§øú.º3Ö0ÆÕ±Ò¬¼ßÁpù^¸§8ÙF½Q…ñ0¶fÆÞ­¾F‰GØ><GU¡ÑÎN^«T)(H$3¹_Ò¦ÏÑ5cá£×ÛÌ_è¾pSÆs© B|ÄË(xÇ[KÛ}g]i­`Xa#<S«NÔ§Æ>ÑP;¤ñ^«%ôNÙ‰y¨ÎÜóP^gÕ5[Ã$ß­åïcµ·Á3¬&|¿CgÊÚ#)~íg3S½ÁgUš#¢é—Êã úZÝþŽJ#<§e—Êè­Bˆ~üm8ùw™çÓçµo|yO]fäåß:„Ò"ºÞºÁ]sit{p­#ÓÓk$ùwöíØ@#)f2 ôfinMdf¡[ Ã#Ö¸EŽ5·A¤ ðih"àBÙ›^ª²¥¸Ù*¢ðЛ <f›Ê½V%'­»C¶”?ŒCNsæõ5óí±m³†ë¢Õ˽C¼Ì;wìdp—Mh¸³'²iÕôFÇž´n'Ü}ûóÐÎOD;„Ëþ~0o¬ùùGZäNwÏŠšC^y>­ú?[—…—³¸Œ’8Óab“F¤ñxòrYç1“b63&Ûwß™-Ó>/hââ%¶¶Û>g^øÈ›‡ØÐÛµ®«Üì¼[îF¬wg¢â¶o6Åó72a¶Ù'ØÂ)KïY>{é;t['6Ĥ„&ëJc~Gva­ê‹Ë‰ÆYA-§™ =¦!ƒB#)oa¹ö@ÂJ.P¼úÙú+vo~8<3bå:K}½ÏB®w(Íä»eUÁ"ËÖÎfcÊÚñV—Çx'>mý“ŒÛp$oGÆ€]ª Z‹{f‰(žÓôÅ@ìR5ûæv66Ìg_®qjŽ#Ç|KÀ6#4=ºG„¢8+Ha¾§&–­Âä 7¤`#¡EßðàrÊåýp?¿ïlÕQƒfû£ÉVU#›N'æf>ù œ|¶‘·±C˜Fu,>‹MÎ í¦m›NÜNÔ­—¹cÑK»±i)²)Ä÷"Í<Lb.*(hàvsºiðoO¡$—¢?Éö9ñÇrVŽ§+irÑ¿Õv®Ý﻾ÐðFœ¿ °¥KW~dKeãåæm;c}¦DÒË0÷ÚÈ—#o,ÓPa°Éü&|3MáUáÇmñÕq³d}¶&”Á€št©aÝÌ–x6·ÆÐâ«Ë#4#<[ÒÇ]¿nü\#a´tr÷(‚y×-»Š ÒÂ& :càbá\aêöë/SÅëÓyŠn·£éW20Åm#<òá‰Ø‚"ÆvvÙ¦r ó 2ÙœQŽ•#4ÌàÅVkâ,|¶x3S¬èCÛC„C2ÑËÆðÇÁá Êô{ˆy=æøŸ—\èï^ûqßC0#=¦<Z›Jì7yӶˋKŸÖ“ÉgÜaζº¶¹Ïø‹AIøê³>hAÒB^ò‚UЃǗð›EúÖtœ#)ë9 ñÙû†Ì^òVá•gO’t¼¹Vç-¥ê­cëÖiHŽ`°7M奓aéhK+p…–}…„Òèµ®³y#< %÷Rs`Ï·–zo±.k ²`V«Ìß°<Z$¶ÙÉG……âxŽ#<ý-få±0°yöCQY‹VN¹µrÜÑUÅDÆÁ:Qcö=[ ÔÅÕƾG¬5.‰íe•EÏßbF»ëvÁtóÙmZD%,ó®epºV?Üby•ÂúÅN*û«¦en7ê«ðÎ-¡·Âß}Þ> 7ßÆq¸¾5W}:?E4b©^IÅ%vë,kœÜ³¾Ù|uÒ*6…[íuÖóCµñÎy˜BºòÁ³Uó$ãPäÖ~Ø~ž™æöé|Ö[£.gà&<ZöØ×Ìc†1†*Eñ¦¢çÃVÅ~ŽÔ\ÄB—•{—#<ÏÛ’íÕõ¯‡qÚ©°vàú_o`7•ÁÁçÍ«çb Gƒ&Ø:R4²bË ® GÊ'¹ôò:.YG¥ToãdzW_3Ç­qüsšÄ]zH•¿…ºSIG‚Úª)"—<I^·þÌ}5øøäÇáÛ™Æf~¢åd‰¾õž/ÃU>V Æ€(^7ª‹/ÎÉ>FÅt«}TùD†¾k'T_Ç+Ÿ†.¯nÌìæ"Ü$æ³WG`x|•-£šÍ—ßm˜ƒ¾|vZ ò嶃ˆŒ?Ç·0'“¨ÄõòàË„óš3o÷x@wSéÇɲ8Öa†W…#45#4úød&$dd¯³û^XÛŠÖGó’k};ú)Žýâ%yò¸{æºWE(–ò¨ó5cØ^ä7®½m#4-"y÷B..yz ~~½M³J{&è•åe]±ßº.ŽbsÓRb)¼‡3/#)ÇLXYc<÷ê²yØê Rû™ üV ˜°°Ês(lw[hwlt˜1˜4î"­Ì'Áãìˆèl7L¾™<ït“½D[Ùq8\e³Á³lX{]ãô뉜۹Ýæ£\9ì#4†×B­‘SGÒÇ<=pv>L£:oº"•®+ÒpïgEÄÙŒ\¬d󞣠á¢c è cVTd³É#)þó¼,ü•D3­º¾xãAŒž_öhé±’ƒdxteÇ#)ÇCÅþµÚä´î—{à“¿¶!¢k¸IµrZ]ÚxUöÛ|–ÙÄïXn¾»ãÝÓ:“;íÅIhº wò¾' 挥ÍÓÙø»×HápÜLÝÑñsUòƒœ~v@a#46¿yªC¹aþ6x@OAHó®…¶ÍVfk¾Ü‹Ú­!cN±²:iW¤´¹8-’².…cÎà$v:=—²'·ÞTM&^(¿—Lñ‰»(ZPñàþýó×CtB˜¨{,°±i2)„y!XÁüVøY|m/sÍ™\o7Iî«bt^²Ñã7KE,-~~ØL¼P!eME”9ƒî¤£¨t)ã'Qú4¸DAëÌW…âtpŽmÁñþpX2-y EÓÓ K¬j<Mí8QÛ°”Q¦"­Lñ{;´Kî`´t\´±û _U#<"dÌ¡¹ÍÍŽüå'¿¥ÐÅF|³é•xµ`NGÒ@Nl&˜kר¤ÒK1Ø%²Ì¹ã;yà>;èšœ’ÞK"(ƒ’Gàs¬©daŒÍ¤ú¹ë1·?_Ž5÷g;@ŸžLS˜ƒ¦nr{äŒ~J £ùJ±xÁ«BØš#)ý¹Ù[ì„pb*ÛcšAÎó~¯£§8ÖÆ.m<8 Ë!`À0­iXÙlšJ¸«¤BèZÖ³A„¹…J7!H‰0Q‚×uÒéÓc€µòY)ZÉÌ;§W\FP£ æ´<–­²ª‘W~Ùº.[¸4åI9„ˆh-œôÔ9<*˜[mÔ±ƒ«Ã³N€‘šáY5•Â;«]n]Ö+?UÏ¢Â;IS|å‰RòkZ«£Öë%ÆSÜÑïâ;‘›Ñ42fbYøG¥7.çž7Â@*2¾u…ĺ‘#4 gL&G07|W“ÓùF:EŸŠÆœXú)Ù7³Á2s ™Eí~UÓü£´WÊ~=þ\ÙžGåaîLZº™ð|rð7q†øƒK¢÷îÂçÙ Ì壽Êg)Pw„ÝÒrv6Tn²›'ª*üô9õƒ‰²g\Æ­¯æ9螢Ò]is¥Ã‹–÷‡3…:Z°Ê¢U˜#Ê.Ò·›Œ#4öNÚ=JГ'j» (QÂgW#ܽ\Áâ¡yÔš:üqò²Îëïñ^LŽ.wÂ?EF',Jÿv]¹>áÊX9~`<=BŸÖöÏ—iÀxNÞé˜Çƒ¾pZuà:Ç?EÄî9à/@·~_e•,D$-*âõØÇo¶X¿`skÅfk¿4Gâ³g1#<°æ5ð-c¾ŽT¶Ôù弜ÁôÛé&Ùõ ú/oŸóôæô²îî^]È&ç§^µ8¿Üó&›7,²»$ƒÓc(°óš\Væ[NÚZ`‘ºÎëfH"Ä…ï¿nd×CÏåµÇ±ó_(¡×éþ©Ú’l$ÄQäÓîF¹–U7)Ó^]»>k®m¥P/Xsªs?³È'÷!§›§ä俶Ÿ ;ÌŪ>SªŒ{{‰²Gºå¨§«Ð0},‰µØRÈUÊ–Å >¦dGG ÀÍz»¶§é劗׎%8SµLm×ÙæhÕø8à·»Y“nô˜•ú»ËŃ˜4WljÃ-,%³âªf<:LÊ ×åÐâ5|ä"_íË.µèØ=ËÆÂË¢ÖáJàáMÍmáÏt£døŒä'5V7v7]TÙi:鹋¿.gA§ðÉ⫵Åë=§{O%¦ ‹†Ô:D&”/·ÝµŸm°Ùõ¢ÃG»äøUÇ„j¥úÅ¥ê&”ªÞ×e…¼)³Ÿ(/7ƒ£Üæ»ùœ?MûâøÎY0nÆLA®1“wc¡K0ìV*„ß/€¾*”óØÎŒTÎþ¨5ïË•æ(H\‰7°Q¥1Q£s“yÖrÝËç ßË)`%ž3L›oÁ;37T#4ølìë8ÁP™z?±Çf©3¶u-IDlÛ7Ñ ršE˜ËKÞ_ÞS>ÊjµžÙô·uy]ÑÝÇ…ÛŽ¶S9Ò®Kà¯åC'‡êìjN#8þ9ˆˆ›Hb284Ù±í€?ÚD6Îøazèi¹Ù௞îhDXLÛ'Vâ¸ïXÛŽ#<¢=¸óŽk¸z©±@¨`ö½G:^#³ßÓ×k§ÑÃû¹…n‡W¯„òÆeCÄgÐRøQ…yv2uëxµÚ‰m-P^SÂÕLÏ;ï­úá €íFæ@‚d ùuö©Îíî¥CqT÷³ª\œ—r—¹AJ”‡EZjæ9 EÜZÓHÕ k̶͸‘`kÞÔÑ‹Âë€û¿q}þÕÚöÑ¢cp•‰Kýï𾪩ྀX§ISD]‚íÛô OèÍÃìÑ]¦@ÕÒáç.UÆQ¦Ç€©®aÑZª½?¯ßŸ_â·ÑÛ;uÊ¢sû½» .¯åï×Ò_•õ‘«U§/Ö}N­u„±É¹e¢û^Ž!zU…óºè‰c·sÞÀ8o8Éh«² z•È8ןÂÉ' &ùŒÀ«•êæ|ô–Sep‚óÎÅ r3Óy«ni8ºbĦ—õƒ¡Dé; æiê¹Ó1´b0èé}Ö‹D$ìP)»4²ä)¦Å#<_ÆùçtAd^¸`á‹gl¦ô…Ôš¸‰«IÖ—°z½rQ„Ÿêö¯ s˜ÎÚ1·Ai ð!ø:;_|4<c´K’T,;ç7¼Íx•~nÊ'eЋäÃv›%%|#qt!k’°SMU"Ï‘XVLrØð¿{Þó“7£†2÷ùòä¶Á#)w=ì„O¬a‹Ùºô€•S´ÿ}} ºGyF±øÆzG­Ð£Ç$r½Í,“ØÍ›”J|9®~™DJµt].x+«íXØÊ`Ô¡f6[žÊ|2ÅÕDµ`«ÖªA¾æŒ3óÑÏ‘ø¢æ\g*y çÌ}fà¤s—iß>žþ¨“­·ÃžÊÔ-žq#<OÞ)¶tç²c *ª§ŸhŽßÄk¾fSDÔ«ïlƒ=µüÿ#«æ¶I2…Ê,Ç zÝfa÷õyeáò#4'ÂáD­¬V0rX|t77t¹^Ÿ%– ÔP.eAú³>U‘-!#)\ZzѽÍ׃Yf…^¨ã×µÙù+Ûú?)öT…€Cœ;²J§ƒPŒ—[—'_âèîV:9uºãGCËh{Z]ßV]ÖÅÓ T{´BB`SBüµp“]„.{¥“ËVAÂ`ŒÓ‚mÕó>¤6q¡‚WA× ;t0£¦ƒ–;ô/7ŽSË®¯i–å âê8 •Êªfþóù‘Ä‹¦ 0”óMäŽ#(z²Ä_4iC V”ã½]³>‡yù!#J»cŒÔ*Y^ú´øaðÊ.}#Q6謃žÎkOç‡cg÷LKJg7ëšåß¾Æ#<û?cŽÜm]HÒ!&˜5{É5Þ1æªï/ëyûõãœrné¾åçåÝGkÚ;#uÝx ¶×„3U¨Õ)·(âH!j°I‡Ù½ê4“¥ÃUÑ6¤øÝ4³ ÚÙP3c‹˜ñ@êи6½£ù}Î_m‰äè*ã<}3ã³V?±¤ [1"ÖN/Ô÷Ü+&‰!ÅiÆœpßZêε›âå8:âñXôœ™V8ϾÅ>O– ÙîLÃïCدGm€»CÆÇCÔo©UW¸%0¶`q/¹Ï–N„ Êõ1z”Ю‡&®#4 ñÔ–iKÛ)É÷AVŽ¦‚g³ËÍW>¾=WÏcªœ³‘éŽ6(¢Ã«Æò™Ó¹¬j'‹‡ŠÒysëèý\|õ»ç:ÑN»û¿º®¶ý5ýF”Òuu‡ôÚÃ÷‹-ØåøÙÈ9wÙZúÖyhÆ+ÄÄ€°I©±D8ÅÍŒ%WOé”Hž:ä‰E“Jàdù í|0a×|ùv‰#Œ%oél£¿KÖ9—<S5NorèÁÓE·L4áùP«*í)Pöo9-çÖL"%$¾#<ïs¡+œ™’>ÊÕ>o`wÌμÅG«AŠð‚çÓ¾þ×Îùóòrd—´JJ—àÂkõôÌþ'M iqφîq>â¡áÛ„bØÞ¨¡#j£d:VË+db5RÅ@­UA.ÕɺHà‡ÏIðë~JO”¨FusÑ9JºIÞåýB­^.a@:àëíîœxœçoŽRŽñ¢Ì‡ÓÚbo˜·²7ŒsÒ}Yêµâ£³×mlE²ÈMÉ NÅ& *Œ#4|uCT×tè‚!,~Yãa‘=øir÷8x 7‹oH¶!š1LJ`cZ* çO9Fã·–ï"M.ÉÜòæâ²L[ZáçœôÁ5 Á…qÚx\ƶt:D#~·µGKÄ\ RôÔ^JlJNþ#<&G¯éÚH;¼ŸC6N¸¼Âü±^~»÷ôßnX~ŽÂŠ{ý/ðBÿ^Õ9©L~ý­¤iô™jÖe:\©Î½XÎmÛÙ5À챋—ò ¢Ü•q3,ž3Œš¹®´ØÉc×@1¡«#)‘~9£ ¼CÀë¿Žåé·>Zº© QP ƒÏu±áºôMgÏÂ1'›ôõŸ> IÆt#)EÙ+Ú-FzûnªØÓ¹ÿé»_MÀ(ÿg}Ÿ.»‚vóm¾ÛîM=Š‚`ÅÙpÌd){"’úÝi_Çå|¸¯p×ÃÁCÁÒ¨²vÇdÞñ·0OSÍÒ)w©Tó*vPM&Úôð‹Ý¿ ¤6ëO5®‡7À~1¯´ÜÊgMp#<ðä)Ø{¸|ׄ÷u¯"Õž?½à¦ÀáÛYnz!Ï‹1w(m*~-bÃ䀌eWìS±ù#<Ò wJŽ>“p"j¢ÒéÂçŒ\¦Ka4­z!‰,æV ŽÓ‘wrÞm1‡y›4«íf?(¹l¼€£(è2Ž›¶¤=ü#÷?cÚiµ+kiʸâY[½‡EwJK‚ŽáŸ¥Ù !®Ä ñ'EÃ.Í¡,ÐÓ1JÍ©WÚ';jüñ뺄 &CïîDÓR};å;Vá™°ƒ(Yû=,8¯œlÌÇݯ’è <ȹ|¾{ëû;¼¾'«OéŸKJ÷òñ(#)`SÖ©ôzÔ§÷§êOÙöåìŠ hFà9ìág0ûç U…¶8;qø+ª&n#<R^Ï3ù<5†_z{ƒSø}œìJˆ¾ž»ÁƒHX-Qmó7ã¬%R|ÿ¸úY4 ZQU$×5Óó#4{n®¢Ä¤Z“HŒ‹ Pp3Jû8N•v©êbsr%àˆ…˜šô Õ p¨×ø½9DŒÜàž»íÇ>—3óºÇdϾ´M‡Ä™ìäoI$ƒ"±d#$C$=ä2†šw–ÈÝx ß)%(¤**¿®Ôå p)#42T¿M*Ç Œ`Â?TòY¹1ïv6£”•¹0ÏØRïRs2'z€ú%?FY„ÁE,@žûÚ>Š“‰GH‘/®×¼?Bò¿Àsi 1b #<¢JÌ«¯ÈŸ*ÒËYŠéP¥Bš`1ª¡¹SCü'»þ…£•÷ï60EPnr·ÖˆÃKšè#)1Ë벨ôÈm«ppxßÞ nvù»®ÕS[•¹‚p€¥Ó/—ÕMß9}A†¼ÌD#Ûç/·X¨ÉÖ`¶t;`v|üþ;8hiM—)PH×}Ÿ07ò#9yFãü[÷ùÙ?7Э®…Šz¬nBcžciŽª‘06$»º›kHsz-sÃÊ2¨ϼK}—sËõeRîlŽBþsK¦{=›b”¥2&-=e—å!òÎL0âB'êýjhG¥ maµ:H9l›æY‹=ª5GR'üƒ‡²cd‘)õNÎB7%§ZÑ^ U˜-â4iÃ_Ò¿v¯oÞJ¹ž ‰V„¢Ä°“­¡ÉÏÌååúÝ–qèÊ¿pãÿª}¢~¼›ÎœôYŽ‡Ö–¾ï¡<)Fô7Ã@~Ïõ:Ïñš‹ý‘µ٤#)ì0ÝG‡- y!#<ð¿îöOœ3RÎiÞÍ#4Ò‹¢‚™¿Ò§W±Š`~o؇ބä™J`sº!­PTP7|ÿËmq#<ˆÀ΀åj«ùûxAUÿ1éDQFK"ÈHžÍô*c;â° !Z;‹9D)ö´#4šh5'Ñf=ê%gÐÔ`tyåIhCw\ºZÄItƒ âx #45ܹ)ÕGUµ%ƒÝ\‹ëˆc#)×úùi¯[•¬§OñÒÚ±Mäôxþünso¯!0ÀfÊ1ù/X~N<žÎÑ”$>=ïh$cwèÝ7ôP¿±Ç öš¾à€G»¶ò0ÙÔ;!6^9Ó¯„:*ˆ¤I¯s„·«mŸö?ÛòæZû=ROg‰0ïãGr8ó?næ‹)VkwÜ&&õE˜° C«Ñ.iT1™B)¤[%ï<´¦“v#4Щ}í·{~]ôúá+Òõò¬3v‰–TßΡ)Uøç[·]CO]góu9Ê5”ThT ç“|"ßœöEþËéf¢äO ¿u2/’H8)}ß<}g®;iÉ—›¤ˆwؘùæWs#<ï=/‡vBä[DFî{ûóÀJ9uÿN¹Žð„óôë4œæêÀôsDÛ‚úÈ­&ž#<©±~¶ç­á®#4FåÔSÔÞôLDçê[k/wBñÞç2 GW‹Ï·…õX“ã\‰*×UlÇ Ûºe#4f؃‡{gKcm³®ºÌ^òwulôªÆZquuŒHí@hgo«@†ÌÙC@Ü'P¢¸¢ £mìI×øê^cÄéûâ*´¥B«ÈÀ:Vëî‹õ­tuCZïÙ²ˆÍš3¡‚@ë…Ž!Q”ˆMèX³TVPxSŸ»¼q³ ù>N׸Ñp#)_\z/wí€<g‹£ußKºäàÂî.#<oz½Øó‚  ]¾/N:|Ì#<T‰Ëâ$¨$šnkïJH©ç‚%Ac×KnÛ­9FK’‹-~v6'êY×?ŸvæØ„Zg<>Î.è°§šãõ”8°`Ø´±³MzÔ¸š â|çµó›ñ£é6„6‡æg©ê3çdÕ/­Õ$IªõªTNj’ŽµÏ|YHaßJÿ9Û«Ø»ÞuëéNœRºº©âX[ŒtæŠ;R1  €ˆe#<¯hB=]ý~üV;3qÕ!ÜbX8Î|qÇÆ÷êç?ä“y3zlÑàý¦‚Nâcxü¥NÉÇ£þrqõ&2¾˜!áC?ø3ªMé/Åõ¨Ç#øZnÈ×ø©µ^Ö<]£Ò-@&œ9›PxK«ÂŽâ²'~ËÇ“²Íýçϼ1}™Útá£Íªä òm‚g]%¬$%r±É\4Ià øºoˆôõùFx€ò4 Üä–P×ß´;ÊSL9† ´z.æ¡/Ó¥Ó Òèß̬`Û¢‚Ô§¯Âöy7IºU(¹.Pë1ùâ hJ´9HÚ埉&xùÞ:fo7"ÍÊh‹øFRH¢ˆ¢ZbüØÃû$þqˆnÂç‡ÕýÂݹ-õ'ª1Í-v7Cåì}•1½\$HCL© åÛ‡hlËö*¹Õ!ý=ñQAÝ¥Žºl‰ØäPÛEÈ m(‰ïCXÀg“Ãnh‰¸#4ï¡0|ÙýX0¢at¥Í³Êaò’ìl™æ‚– ‚‹#)¬›™3-Ï`KHt­¨Bà-Gêæs"’ —vøÅ ØQ$Ê<J¡ÕŽç x#_]Î^Ý0¡DðÊ;ŠlD¯8[Â…–÷¹ˆ;˜`œæù¸p;a°Jº´¸7ù>ÂÇçÊëš“7\<fe‘Ý“i‚8”o–:×)#<ãЩF‹*ò½¯Ñ¸FºYç¾»H,èxóñ4ý#ª?f¨þ{(!†z‡—nóNÞqCûà0! 4¿«(X ‹kcu¯Ñ[‚ ÄøŽÅ@J$™J¬4õ½N{ÊÌNëÎÚ_~+HGoõüuãŽ\¬7Ùhâ#4ìR!#4€š#)¡2׺/ã¡š” paM3›"Ïê[l¼ùHc5°9Û–dœàè}ß?£‚:!}ÙØI-u·<G˜^¨Šp2#H˜#<¥KϬó±ý®ñòƒ/ŠÓO;S&ž‰)-!/¢+ÁË Å<›Ãû#©‚ãÕÚ;0æ¸ñ¾0ç~û·£¢>i¦XµŽï׎ØHN—»w7E¦)™ ÊèMfze–j~ijUod@ÙR˜@-€ #!C;{y`†ÈŽ.)jv]AÄmua,´ìÚ¹ÃEêUL¬,Ö¢(ÇrO'ßÆO–Ÿœ»ä’Òs=¹ˆäæ_PìáŠøëlb)ÒìÞ¾°&G¦+ò+#4¼wÓ×´ÓÒ²Hí4ù)H'!àç2ÆÍýÇÎ 3®=ß=5¤#\T<è±Ö`Rž/#µUÑ^¾MÝBØÁÒMe¿$+Ÿ<ûðBœ*z.öÀã=ä=PsØùµ–M场ÓT)¨%üvŽf¬ï¿>ž 1$¹µ3s/ðšT[¶0B™æàFdÊÃm¢” %rßKhì7lÔ#<k,ؤáâ™Ln^OIÀ{±pí±V{•FpR² „FjúX/#<·f%b¸(Ìa1Iµ´ÏÒ!ýkibïQï—¼Ü#<]ÛÞ¹_¦×õ;Xð}gšÅ$'ÖB”pe ]{³qUTAøy„“–4°A¸ê"èÆäG\I# r.èÚ‰•ãÉ'Pt5*p Átƒ¦ÔR^ŠšH>ÂuÁqgǤ®TÝ\˜‹#yŸØ˜ëO…}ãUWM©óÆÙ©¶2t¨-‡§£#4ô2ÿ‰SrS¤¶ ç:Î ¢ãº}¼üKø@§Ï~Fþ¸Xì0Ú¥I«s+×ÒƒÈÆéªÛѲ±Bnû—Ûü‚ÚÔú2N’@‡èáuH‡bŒY ­¦4¥˜:PJÜ1)WÒÌžGn)ýÌ ©}P âi̓Ù0–k\çÓêW»2#)âññïLyON”&6¨‹Ñ`=Ž®Ö½z¡¼ ¨ØÓëÌYêÄ<NÖ§;؆¨Õ1ïÒܶgeêºÕ›Ã¨éÙ‘C†6{#<JÅMšX´‘>5n˜N½ÛMxŽœå0Õyó6â&‰à½ç…ò_aYGílp8,±= ~:H‚O,GfθÍ#+«P•Ø@dÍQ5+$#4À|3R6_’‡Õ±)yOç!\ÈúÞŽ®¸ý¶ºci!TͽÉ^çXyW¹“Ö„ÿ!UAU#^æQ>„þVƒÕ¶ƒàB:¢JÿDÿ¬ÿ•\“÷Ƨ#)Ðõœ##<bÙUÊåÛº¡©D.ì, …2JU`ú¿Çë}Ê©¦{R¤#4¿Ñœ²ë‹úª¡ýÐCüy#<ƒŒD¯Aô=r¤ü¶óV9BeåAUOŽj þè²}$üGÃáØ{ôz÷·×ì !ÿOZ§Ç>¼±QUHIýètBЃûI3ûP4#)qø~ ¾ä÷‰'Åëõ3Ä8úqdÀZS}৮aB ±ð!@{××mÍñN .ø;7r‚„ßK\ã@áæÝŸo.!p¨æjãGV¹NÈ=S$ú¢R«šîö¡ d¿#<0Ay²˜(ŒCÇ­t|âäºaƒðãÍÁŽŽ\u9¯>uê"'^|?&Ì{Úa¢T:Çšrpû:xãˆç¸ÞËÛÙdçtÚ!™ÌT‚€xŒ3ʽörBù ÒÊÔhû7CÀëÙ⨦ٞÝì¸îäÝJÆJ’„þ.RèǬ—ŽS§‹|#úgâ~ÎÝù¼§¬CA9`X°;kÍÞ=}¹l…Ç”í±­ï§·¥äé½^6Î…g‰™îa¡““L´¡W~¶;PfgÞ ‰V× ¯´r BÅîÛªé?DhúŠôÐQåæòþû 15EźU,6¯TM„³ãpb+ª÷p¯ÆìÑ#4XhÀ¨0UüCë.Váøæ|ÐþĦW,øþy™9|ü}Yüž¡ÌO‘Q(¾ÀzY#)8Ÿå²P7ò~²”?²æžlªö.ôA`Bßdx#)|‡—ñ;BË–Ez¾M˜µpUî(3@»”BÌ=,AMP²#‹}9%#<²)\@‹@é )MÐ9èÀŒ³‰­°6环D Bé#4YBd—!$àÃK–GRÛ(XX¢æRü¬ €6¡3[ì/ÑÀ0#)°Ѳh”g¸]âçc&°ÈÛ€°,8$K(˜`Á‘Ù…Pd„&ÇËDAPC8«üÅ9F¤=$—|FSý÷íÕâ=E,Ë"#4$94è”Ó)Ðï^áP9éï‡F_×Ò¯õàÐãß$ÌVù1A·òBådÙÙ§ørðI,8â ÑAb"É qf' '«áhö†¡Ó[¿ì~¦Uf*µ£e!߃0ݦ¼kZ£Ž,1Øéèô-üuœ×–ÑáI’»ÔŸ›©{AÃßô9¶s¬’´ÖäW ä#)“``ÛšÅ{YÒ¹ïÙQ ¾•£u“rüÌ »¡¤›Dj ‘BéÛµ×uw.™·[»ÒŒ^êßp‡#4뉻ìÚîMY!nSTìßt¨pö'¶«\⧷·o+Ž¸Ê¿Ë' âŠxÀÿC¤,{Ï ³d®©gÍZˆðœh¦Æ6Æi£ë½Š)ÙÃàq8q<žëQÈÔ•—.Œº©82w³FjVXzê¿ÐãÒ©òiŠ`Eˆ7DùÔƒu~‡¬Ã`ðl<Z vTT¶ïÇjÅŽ—\ÝvÞéê蔉#)žkÅøP[=ÏJ= Zh}¦5?ÞxC‚~´{2^áÌU“÷†³ådÇ&…ïîMf´®žbÄ?LC×.#4ÈÂç¡`z½˜I,ЪTdX ˆ¢ØÚ)šT³ZfJbÇé{øΘ8*ª‘¦(-1*¡éÛ¯³Y˜Lªj ÷»qʦsi6À£c“¬6˜Ø$]ˆD5fÞÂÝ6væẆD„À\uEPbCTëæ–2‚ŠÁ+ĺv‹= @ZDý'ƒmìykÁHÉ餉N², ¤"¼ÍfÚ#lË¥t‰«AÄ.°ÙÄÙjD´&‚l.‰„I$3§3Uñ+?fðLΉŠ‡ˆ;S´Î•Â¿­ÔL<  ~ „“L]8ЄˆPvVòhÀu*Q²Î9éc†…Úto¼ò÷熨|“g.õø™à2‡5ìy`™å[Èñ8@˃ŠàqHž‡Œ$<Âñ»‰ìˆe ²GUKPkDyÖd.r&?~ ñKƒã¶×‡~¾ç†JŠ'‘Zvêm JÖsOòñ¢«™Zà‰É¹ŠÉC#)•HæAS6‚°€,ˆ6”1Zô­åWe¼Ö¼ÛUʲmW0ibF ¦Ñ!>‡dÌDÆ›@6“è»rˆÝ|íîÝm͵ù¤Ù)4`¯¬è¤ÙYAs\HŠ3\1.jÍ]ÁÖ#)yq ¤¡ºaëtív#)ÿ[YÒ,dseÊ9䇾òÇEVT«¸gÅ“çqŸQÛ»¡x€l‰-C_!>;ä[ø× Ò¾w(C¼¥ñëçY0§ÒV|ªhg”ùy\N’Ni`¨‘„‹"!"ç °lK"Î>ç|•š8;: S×Ç`É­ð+Âë1dò=›„.yˆdRüH2®8…˳pz¡Ù\­±A°ŒÒ#)¯¶HÅ­ó…ç¡ç&ßî­`ì'Ù_†ÅIã ˜Gfl|ó{9¬JŸšrÅhÐFháT@#<€´T‘MŠÛ[ë}¢Z5E¯Ÿè^E$‘@‘!TƒKåmA5~¼}´N9Öþ Ù;z“ÀZªXÕkóadÚ§©D g]ªLŒ<QM(³8´ØpÖeŽüùó·©î ðÛÆ*‡©FPì4&éb†à`CDœ1¿¨Ž™âEgºÕ¼Ž}iµ"éš`@  "¥ö`K#4LYD¡ó¯:²…(Yt¨6Ã`@ ÔŸVÌ9mã¨C¼çÈ}»”7\´ÐxF1$ÄkµéC}¥z$W#¥ï.”æ¬Tí×+Ë4ãÊóí¢ƒ¥©´Â‘`²¤J¼}t4mƒ)À¢R©¤R „ØÕ##4œùgpDíãYûöÌ0Ò°,†Ñˆl`ïÞ†œ*5 OIÕ†ÌØS#)£c\üf†?¹Ÿ=—ëÅ<«¯yg¥”"*jªóòH|)dX$(*Ëô(¬1¨IÝ.k©pÏž½y¼÷.ÍWwJl– º1³ÍÚKòdts$ìâv¨¢§lÆü(ÞŸ*Ê3 ̺8ˆl)ðÅ62fQ1®õÅÛrsñÖ—¹¬õ¹é)˜áʧ*Jˆ{‡q `ÎŽžo©š¤”ÀiDÆtÒ· È2);ž£ &Í%|Ól1‘«»™6<ÐÃ;¸œ_$Màéªx±m¼ÒÇ›LŽÐ5ÊwoÕ¬7#4ÍgnÝç»å5Zó³{š¡$˜NÀ?#<{™BlÄÄÅëéóaaǜӜûÛ¦ýΤ߬é,7¿ËΖtúHƒLf™Ïêôðš<ÜÉ$`¹‘°1šQuïÍÉ.÷Ç}G¥ïç†Þï¿cO¶¡Ý¿:vÉ,ã®'ˆ¢½ÎC[òŠQöíWTñ·¤JdÅ-õΞ0œ>»âpzt¸ò6óŽ™/ïà(j˜HÆÌãO`pöþYôoªiÉ]åMìxC¢, tæÏà|´ðÄó‹Í^ÎY:Ý$nG$q’6= ^©WÝ•íš*54²Úâ’¼¿O»"…ÐRúšb-æé±c…B^´Qõ÷<kÆp½YëÙE&œ†ã‰ž~e¤“ð»Ø2ÚOŸÕÅͲžuNÈ°<Þš by#á‹£8¥•]“Nëömݦ³LáUš¨‚´.Ñe46ZR,dó8[€làæ&—‚ÆXoz¿pèe–p¾ ¶…òD"C¶sÜíÊJ—wÐïQzèß!jRí¹¿­«e9®¤=‰ñ]ö²NÙEo'v.ÙçOQŽ4QÁJ#< Q`¹’Q¬KCƒ%*ˆwΙ+Ž¼¤¸švÝsí,üà¿•_K®ƒ+º9*÷Ç|f<tòûé\ökªÚbÍ‹™÷¾o}å™ùe_ÉÊv>ײ:Ó1O3ÖŜΆÞ8!ŽöPdXfoÃ#)=OŠØÄ•N¤,ío²…÷²ë\ñaìÐÁ‘ÐØÀð~#4¤B9ƒ·ÅþwQð¡ÿJ¢z×0ÎŽˆsd±ùÌÈ Ì˧¾o'žÌWðÏ*·xW˜\8‡z$<´œÙO#)(#4•‹ˆh4ºg©S€™#ßR¨V¥;-À»²„d=šrPQAe“:“…n³  í½¡7›iÌœ ™¦¬w×ÇåõYÓÃ;sÙ¬6Jª"wQØ(,Uƒ÷5!ª|5¡ñ1߉ …Šª§9¤€{»^t;/·K£ií½¶ñ6YEш÷ngÖ¢y:‡½O¹ÇD ØÛ|‰¸Gwã&Ë’%¨·8~»±,ÖÆD ¼ñË`òx¬RuªgÊÔý¨UV4‚3Hë²g»Ú±Žˆê”$ÃˉúÏ2tÛëŸç™°‡§Coo!Ð3µß5¢Îf-ÉPbPErl!›òZ€4ÄÚnsävA=^c M¦¸ákõu´‡HÌ36šÐà@y?CÖqº‚Žä5i#)Âw¤ž,-à“›âw‰`œº3‰ÙÛ{`¶Ðº;‘M AìC2.û\•t0¹$Ú#<Á6ÛEIñº¨ÒkMŒÖÕX¤-vÃIub¥B©FgC[±4³ÃÙ18Ï‘]EHuÁÈâ\Á"¨=í¸”¥€ÙŒ§OÁëµ?u‚å°fžžŒ4Ûk®)á~äïm@ñëÅœ™šð¡ébÐéOI— Ð䇅Í'J¯™5e¨‹¯}%‰ÂXˆ»KѤ)Äà>¿ ê®Ë+·Ë£ÈŒÅÜéÈàë°>é%J“Bn´<Dù•à5‰¡c&q:øŽöû8l6¼àxá®Bˆ‹§Ì9sOÅçÇ´¨dQÜCÓ£Z§¯L<'ÉžŒ5u`´¬%1hワÓw-ÝûpÍ;ˆ&íÔ!×û8\)ò€)Ûú?#4›'ûölñ{|ÿÑÔ;~”{¬ÿ ¡È‚V…üŸ<Q#4˜Ö¨V¢ya{*¾rYxéÌr1£y®Iöï×øTõ('_ëË5ÞöÙ{ãª&ؘ: Ûó[«Ø žø¾<¡´ï£.p˜ÆÎÛØQ n{Æÿ¹˜æâ– à·‰‡ŒèÃÅ‚²µ{ÆP²z÷¸Ws‚ ‚#4}CGÏR–Ílsq%ý×"¤%ÿsO¯n0ÏV2¤t7×Kükº¾#)‰‰ÔGÊ2Z©1Ûë<õ¯Ö“@D˜”FJ5('?oæªþøwBö«\·è/úîap|“8„#)íÿD8&ì#<AQ$ŠC$—‘2Œ Šy~ÔÜÍk…+01MD[ª|Žb¿Ø88âuKè;™×ý¤8vT0V“ò«””õÅiTz´DI•x˜ýE‚ (ž—`Bzíõ•Ù†Þžm0¦Èxøgd@õ× €¯éŸèÿ?Ýýßš“Ÿðù?Œ‡â_6 þÿôc úÐÍòÜX¾Åf¢ÖLçÂ*+?¾/ü'MS#4#<­êå#4Z¹ÏR{GaÙ jFAý þÅŠ*asûæS8_ÛòÃöêL½-›ñšÙ¶'ÛëËJØZŒðѦ•¨Äb¦¶ÂÓÛ‚šáLJèú#<,åMUCT‰] }Ä8?Óüw'Ã5(¢Üa¶Z÷ø¿q¼Þ’4ÔÅL4„ÒçæoÞ¾~=q}õåéU~iÈþæNl) ?ŽmÒÁ±µ•ožíÿ)édsp:¾¶lÁqO¨3›áôÉÃÒP#<€RJ¡>˜9Òïg´Z'ŸÞf ­I|Ì=ta¿*ŒLþ¸Ö:ØŒ\Ûó½·ˆÞòiHÃTLNûî ñA jè'E#)JgÌ>þ‘¬Oj‰%P`T;Ž\z/ËNÞzjyÖGèÿbò´5ÂýÓ3!~Uê¦Þœ¬ŸPӆ텾€ „ð’^Eöâ,rP†é¤\à¸KEÂ7}' µà¡S¥u6èºØV ¸¨­®aý’×›%Oà‰Ô<Û“JgÈÁ)÷Lø]šz&‘3çeïv/©!pÏ#<¤b‚‹<&,û}ß_eÔº#õ•J[Œ~ŸÄ0N¼@ûR™qe@PDc#)¨JDÙ‰ãõZߦræç]Eb\®ÔÚÓξ.Žn“.f†£RExý6ÈÐñ‰™óq³WØcþèO§èÿ^ ?õoAÑ>lâáXÄD‹'úFåÕ:6EUv–ºõ¢%%­÷²ƒ2íÛY0)úÄäàë½[¹…ÿ@Ùw§¾ÈKžh£˜"hçÐ5¢|Ÿ›Çj   ;T‰ô/3ÇWPºÏg«œõ7oÙät2ôzsÑì$TZÆ"pðßÏÊ wÕ⊹r$]î_ÍNÜödå¨>«õ{ÄLVÁu_sä³=ö³)_ÞÌÎ-µÌÛ”õ ocùÀ!;<µ))#<ç[Øø-žÓñ‡"aÞO”1B„‚H!ˆ"ŠÃ#)WÕ‡ƒåú¼ºƒ1ü¬z–×÷{ýSßíj…Ðuí–Ú1ôý;711<ð>U>’.œG®ßº½òˆÃpxüFÁÀk:´'¥#)Ü÷²dDw;C…ÑÞpþ+õ”$t‘ÿÇ•èïç¼CãsE{91§¢¶ÇÊO ¥ÅñO ahrzÃTÅÁ P[(ÓãïPOÆx×æÌ_¦œ¬u}ž+·×è6ì1·­Án’_ËÌñÐ9+¤/šuøQa:ˆÚÔgì÷C;½[†Ø Õçxëo@óz<©S4Pû½:>]îÓ¨«Ù‘#)AÈõã#4…L[H˯+j9J™@5=Ú•Puí `þe!êŸÍêÛ»d‡Ä2±<üÊ:µßM㣨pt®!äCèV…ØltIû±`#4 ‹ñþÃhÃÊë#)®´÷?S<í/jfŒ‚NPlŸ£âšxs?x#4âœyztâƒRnOîÔÝkÍÍûžqçJ¥åTTKÃÍNýŒ~kÀ£AT8àIæƒdT)ä!<XD/k3ˆ·ä迾HVŸŸUÌ,¡êÝOš²ÁèÈPu:—¦“¬ûBx}±$þCTá aBÍŒvXXy‹•»+Þ 3Ûãåîóv_2¤$‚ˆ¤AÌ™‡;ì&¾œtzêSMuy¼I>“œý~?»Å™§LâºC¡:ù{®t¨Ûµ[vVVSöÆN ˆh9žÁ¤V7óökAí©µ=cÑíñˆˆr{)4Õ-5Ýð¢rìµþ„ã¿— ç¼9ÝÕÝöt‚ Ý”©8dV7Rɸ÷.»¦!@å@,²Õ[–ÕµCÜZOÞɲ~vÑŸã¹sfÑÂâ´Lþß-‰.óÎqîÆ×IF>#<ö B–ëãÄ#ãElfù&¯¨ñמ¥õí•Ùw¢#)‚#<õ`GŒ~ïç¾?~îŠZ^œ6Ì" ˆÃé,Œí)þ#<ñ©²Ënó¨·<­Ý¥ÎŽÉe¬ƒ€R6`)eØŽOÕ:2&:€P‰¶Ã~Å®ûöær΀€%ÎïÇî)ZŒ]½¿+}Á‡ŸGy€%éí5Ú”¤¢j¹ç᪠‡¹õp”’L]Ƴìau%<”WzïÅÁæ³Õ/gH[®ÖñÝüþ¥uÈ?†Ì°=FeE+1³yÓ¾A„– Uɪg­ÆÏvÿ®ÇÇÚÈëå“Ý•³ü??tá{òûðç…ír>ó¤qvs»¶íþ{lÉgAø7¨³¥<[Â{ï»ý1¾'+qfx£;ã¾j©ùÍK¨#ò¾2ƒeò(¯.q'EPEøA_¼paNw’ÙuCúrûË¢ÒGš:Þל]R~þç§Eo{{VrýSãxŒÊ¾ÒÑݧ z3:6Á´YI*HÈ+į~Ö§Àà5‰I¨¡N#<¦‹PÂêÈòº6™£Wáü8©Çªxz¤<=ˤ¥A˜)7‡†¾@Nç¶,ñZ†S·,Ê<XÍE[Þ Ƀp±òé·:¤1µYʪ„‘Ê'©ºô,µ¥vR•{ùV0›rMs"Tž…WBÁ#)ú(Å«¯+ÔU1($ß‚ŽZ<°Àßu¡cÆ2‡BaB” ßæ9ÇÕ…±Ï<Ž{b~NO^Y`¹Ü©°±WÍà)´Õï劘Œˆ…påa+ƒ} Ó:å?èk™ ÞÍ@êñqïFnžnÐÚ?ðU‚Ôj)fe‰ °kHcqÐÅD¯,#<®Â÷ã×g-?ëé™aÒÒø)iËå~X¸¤Î$r®g÷ïQÅ‘Û—"»C¶ÂqztvD:çÒÄû¿]ìi XFmd­¹\1…ê׉¬#4–5ò¸Ò6ÆL$šA²!Õ뛥ûæÇ*XZ’æ1ܱeGˆ3pªÎÎ9¢•ßÛCWžÓ×.(1ŒSª‡ÑüäªóMãÔw}³£1á‰ÜÛÆ sŒŒxÅŸ›“»â¿#<ÌqEø©ÂübŽ½5Çzλ\ꇨ1íé¬Á#4X ¯«ÔW‹¿o,ÀhÅ¿Œúg­íÝö×ôa"X­ïrq‹°p`ÄÞË€¦÷mžÉF¸0bII-.ê7›z—Bìï~x™m‡Ñ:‰vúùüÎÅÁn¦vêSa΄ -#)ç(È…–öˆÓ¾¢¯K½‡é•ÁZܺÞá­TE¢Ø3Hºr“GÏG#”hŸ9~RRHïd#<ZÙA#4åD{!#<ÈÔ9ÿw³ÓU¥'r‘™}ª#4¥Æ°Ä)‡OÞt¢wyÛ1ïßÖ¶vÜåÎ-×óÍ¢hÿ¹´4Ê)ÅÀ±'¨T3Ape-Dæò«¦ak*ˆRR½ZDìÎg¯¹;C¥[Kâˆvuûàû¢sÅ3‡É!Íúý,ü,t—ë Ó»eü¸àÙß«‰D\͈zw”ø;{S×}ªz‹«RSuH“†~öqÕ|¹o¿g¼è¶:Ó¬;±árž§¤“Ãàðؤc/ðwÛG¿´ÎK½;§tSžèJç#4>&ÙÅ;:*‰‘”£GݧOìÄá¤ôáWt­±6Do ´iǤ޺ýtéÌ,:ÿ’’Sùsû¶ˆyÆ^=²Ú»T»¢ C“¿Šë¼åFOçÖj¼ß9Ëö¸ËHô×#<=µì\å,cöâ,zíòò±ÖÉ{žÄrj\nû.XB符:­tøœY(ƒ¦olÅ,.èãÎ×לҭ­Ú‘55„S&ƒ#<4‰‡dƒåûœ‘R’+™#4Ø_jö«Á1 #<ÉEÛïèÚîK#e,,•ìsÍãy7k)£V/Ä{S8 7×ç›.G “·ŽÅÑ÷cĬó÷CÆmÝó1ãâðXë4²Æ,ÓÈ'l7.IåsÇG‡ÇÄrÎœìþœt5™F<JôªÖíšEq€pÅV xß8:1M¨ÑgÉÀPÉȯ±žâ»1Í}ÖV)§Eµâô‡M, æ#4°ã#<½¢—|ß…háܤW˜9;Kœ›çmvãý.âi‡ñ>×oT¿¦è6|¥OÚ@þ¹*£"¦øu³ná8Ž~Þ/ä¦î±îøØ:¬RhÇÝL#4ò/_Y#<ùø(bsí©Þ3Ä×͘<õ§å¯Û¦öÎsék•õµÎŠðÁœfQ|VðôÉ#<ÛØç(®{jÈ –MEÕ¹™jÍ'   w… Oí=ý»x£ì>”¹'Ôv&‡üÄëã.Þ¨iÇúæôò"RØì}ÒOû"Ñ™ƒØõ7²Jqø^‘ù6©¨Ù‘Ù(³•“â0–ÄçíUï}¾¯##<œÔ„#ñç˜ÇÜsFåãòpuÊÇû1ßÛšàH›aá×æ¶ÞTdM±…D4w“Úš: âÎÖ)»a…+äî–÷éV&‡9ŽS¯x†-lò·ú~Òù—#4;;õ!øª:iÛe.â·ty‚·WÖc—m±ÙÀci.GˆøþɺS#<9Øh‹#<¨Q.P¨àÃJÝàº(±ýJ§¹H0êCLG†Q‹ÂvÄ»57!¯«ög×õ]†Ùpy‡úM…¨x¥²þÏ•ÀZñw±ÿˆBe.(xr]æz}8ðÅvÉ®<"åì§ù$Að‚ðåÓzVm6 X'¦c¢t7äîrìùLŸ¤Þ ×±?¦H_^ÜDÿIÕú ç.ÀøÔqÑ4˜9MRì8ñcKÃ̉~Æù³ç¨Hmûغ~Ý¡eÊp…Úyø ÁëˆÔ¸ì“Ä[˜†Í²‚<2ÔÆWr¥,.tñd”GM¼îèæôñÍã¼ç6ʽUuA畉a-ÒÆ:­U”7ÁŽêÛØZe…%1O»'á}Ü(~…Hvo½…goºg­CW€âÃÆÁƉfŒˆž¶ažl<¦Œ…¤õ¡x¢ŸÉÈ8„”$ q#<Dûï`ÔƒAVï w çC9Åv¨éT°{Ì‹ ƒ4ïËG ¹Hûüê+€_PÑP´d m)ùÄDlm#4‰L€FS#).cMRK+hF$Âm(L„›Õe„2ùUCÈ€™Q"òj[OXd}x´íPz¼²£ìåòe`ä8ÿ6ïº ô&ô‚Ôöï&&°ñùëîƒ3”nó؇Ô;8pËžX©wÃ=ù…}î´¡‹?.æ{Q<‡z?¯;äé/¼i|˜Øèt:³ÿF<¼|LJöÿY÷¿QÃge\ýR_¿§ ÌÞOm墄åE§MžÛÚÞb¡Äˆá#)^îgò˜?_ó²0LM;Ä#ìäí#4DŪ0$<#ù.ä,åþ~%"ÇyIüäÉD,}œ<ýUqáó6Ô¢kö“?pÜSøbšÉ@Œëþé_ØÅþÊlíþSýÁæ"mN>©æ÷ÐxÀöÓ{}ÛÆŽI¨ÇyçÅJ%þ?´ý%öÀÙláoø˜Y@ó‘¹Y°ÊrøÙâ«áó.n5û¥ÒŸä5”gõ¬>ßø4½?Ótu>ÿ#)qâ<ÞaüùS˜t˜.Ì1ÛÜ=6|·ßð»’Õææû`ƒ7¤HoÚFrǨoìW>ß³ó|yëÐ…$=!”ŽŽÁqÓ¤ybú§ÕÏœ™Ê%?zž‚74èÐê#<’ 5ŠEF(;=è9ÿGXhÈAzQ2µÌ9#H:Ë­@Â%r¤+—ÚzÙó$’ü!V˜ØA{~?ßìþn?ÍûÇiû˜Ðˆ‡Ãbôrx#4¼PŠ0Gº§Òh Ö(_’Í£×9¹ê÷bu$¢çûr¹ ;žãüLVëÅò>bâX#<SÌà á³â‚³¾åêoÙªÒ ÐÕ=/Ô]ý‡îû:[œ\$5hynXÀr•‘m#)ôj͵7&¼’¹ä·¹@1&; p@$AÀcöe/‚½z †Cª }ûœ-lÔdX²Ä½nf°:`#4Á8Õsz“Wïšt‡æ0Ë÷p«{­è¿7FñòОnuÒNêó—)ü ¨ç¨)”éZl¤7ŽÔçP¥O9#<HÒ;Eï…—`è <‰Ð[¾AÇF­Ö†Š—ñø«…æ’ÝAÝð4G#)°‡k.[ ¡ÙÙñÂÅè#)VE lp²±=iÓzXƒzŒ¹$ÏϲU!ñ Ò£T#< Œø×,¤0µWؤùžì”y]†ï&*¢ÂM'DÆ ŠÆ &=|ÚY¢Û8Ô’ÂÑÖ‰šm&ÂÁ4gËï#4²†²#ߨP¶‡¾¼ñ‹¯Ï£„¤Û[“Aéd×íw4 ™÷õòº­wÝçNXiÿ #ûëïZ09þ¼î#<ds߉èÜéGŒœâý|ÓQEˆ.àtâ%]Þ}¹&VîÍžh ŠÀm3µx8؈ߔ«R’þ<¼­A.[ åbÞ‘8ruwéÎLöXÉqŧùn[™¦m–W\©<ÎýÙ†zÜpœã²¸JŸ‰ÝŽXž›ô._2›5”š÷ó«Sž²¹¥ŽÀÒ~pèHÜ]íßæ ›¡ãF·Ãæ¶×[¢º  `ŒàîNh?n6Ûh0%„èÛîGKò¹•‡%gv"ßLÂ@P§-œ¹9.*#4öÜ¥p[ö>Üz|†Àf¬oÇÈ.ŠoŠ#‡ Ÿ—\ôàÁ×u¦î‹&=‹m¼Go ³-!Ø {$PmËáèŸ"îP]ªŒC¯„[6³H‰„ˆœàAá#<Mõfä3Aç§m4Yçô6¹’eï¯ÓO+$t2õE•*ìã`¯gtþôwÄ„Öî>.5vv=“R•ï{m3Ô²›| ÁƒÂô‘À¥\£ŸÙ#<»rÑWTǃ”6‘šÍ1jëÆ°Ãö_î…þ݉Ô4hx¼¤¸Å´oµ¡ú¿[;—EG [{èôÑw«ãí¼OÊE½Êål˜sy\ðD®gA@LMQ^ìY› ÕGï=0„Ðqœ‡Ó#<#<ÁýØ0äîÓkbÊûúâªq£÷ˆoË]Dz‘=Õ¿ô^¢é ?¥Qô¯Øìì{ë>¤•„ÇuçöÉûç@zc( BŒmp½'{>åyLÃ(ÕA•¶è¥ÞÂ-á²Ý »oÑ;6ÚÙÆ•5ÛouÑÕ¥c*—(¿«¾æÔ›G½9Ói–èز”Ó¦p„ý…j¥ÚKQ‡ª¢÷Ý+Þ‚#ôé‚}fì;cŸ2Ö%láëWˆy7£)®EÔþ „Žfú\œ,¼œ29M¾R±ëRÍ8抪£ÛëˆðZZ-Ï»å_Km߄ꮾšâ^Í»>çHhn<‚i¥-Do©½®.qF}jåÍTb#<q—jË•:þ>,ÍÛÏ¿]_¹.œuõ³/ÝDº×ûG@È3à.Š õÓ>–¦È555‚‰mgi{9ne×8‚ã³L#4Õ.jhÙ…dQlÂŒå]íÀè^Ã;Wjψï·¢'^[)Õ´f¼ÖëE¡ÒZ‚:q§ìož=ý˜ L“Aªgòøyr¤ªü Wù±Cà¡žt~î•&瀶+<Þó¢zë9Fw-¥UTUk²ÖZK Ù °Ñã¯hùis××™*±]z|£š!†ãWgaQ‡ÈÂ2“ãrß<3{J®²i³}k–:n]eûÜt„÷Γ™öê$\Ï$vË=î± i= Ê¡èpQ$¹:0Åh–Ï2“)£@7\¸‡M8ù´n Gâ¨éÐTkK’Ûâ+ë^ÃçmÚ½ÆB°o½¶tyYîvã=™¶ºýüiwèâd HP«—ŽŽŸÕ“mu?Ï1sKø£òÑ™ˆX±¼;#è3lcX8 1°æo[ÙÚ¬ÀÉîiT^>½Åƒ“zXƒ,è l!¢×¨Ù>Qgôó—?y•oæµuZ û3¯ùÿEËøKrקό/åùÚßQww´Œèq3ppü=bí®î¨Sfy}u£Á#4Øpª†£7zQ‡ëg+£®ÜË?á«”!ã«G;L Ò!ßNcð–"<_=¾Hð{ ¼'ï!1põ“"ôvÏ3CËeË»9BnÐ4’G21ï¯vˆÓ*Áp¯¼ûÌ1tΘö!µjè#4’ñºš†òêí¹Lí׈nìÁ\€M¥j#<f@÷Uí-SôÙ¹¿OÎ-pµ#)ƒbƒ´ç¸W@Õ!hi·¯®·^úœA°Õ;­€ø¾?NŽœn¥1P-E´\æÌ40ö9ÀÇL‡(gÌx¼4ý‹rpEËN.æÉfŽøfô÷ÿÔ?Nvþ—Ønâ N<œ£S2yìJúëÞž_>Ëöuµ”K`1¶AÚþ0DÛ—íØR€Þt{<<j<DV‚``#<QÀ*AåôsˆC†³z£ãåÀлûpwT¢ÅUú¹uš¿ÆbB8ÂáßDóuù7oÇ”S ±s›Q9UFÑÃ73…*ü5à.rØ6ªJ¬)|/#¾CcÜŽçmV…µ^Eä*Ð P—9¥·|œšÚ¨´té%_¤öº'A(#)綧œSSÏ„Âkðµ©x½á7Hfû!‡#<Qòà'§—k¼<•Îbh#<¨â¨04­¿@†¼¦ü¸²y›{#)û{À»šðî:XsOFöÞÉÙ/$’¡¨HG²ïÓvìxÜ"ûÚn¨ñu߹Ä(*f.Þö©"H9 r~ÍߊÕç½b‡gA¬¶y¢gÁÇôíƒÍUp/ÆiÓj¶/æÞçä“Ï=ù? ñš‹PIGm“´è)|'*wßvƒ…3wX{0É:· ¯ð ÑïQ²{‚#<¬ ³lP% Z# ¡˜ÂR‡66^R3×-.ÙjÇeàlî~ã ýí¦Wç§{Ib0Ï'Y™”JÉ_LÔRõÌ@^Ùœ£c—A½”¤” ȶlHb¹/¡§»QݪêŽ{‡7ͱW9ç;SVÀ÷@Ú#Èøïsˈ5ïC¤¡A¹Cn¶†Žâ¡Â%4˜Œ¸vŠr€y×_ Ü.Â7‰5ïò¹ÇŠ®>L§\;3·iÚ3x@mã²#4–ÌûRÇo·MòÆüYÐà],Z`°C­…±=çJ ÖÉûÉë~ê•×f;í>vÎ&èƒÒ9†:ú¼™Ó¿Ü§©lZíò~­ÓÚ¥ºIº=»ñ.á;n]\TXú > AÜrØ+#åÔ##)ȯ/*¯¾æºÄ5û(‰ŒdK}VZXʵØéP¿¹ÛDìž‚9#<ss¿r î+Õ%<¹Äï<7ôzáZ*oa¡QBì¡ï+i¶:ÏT9^Ý@ð´ÒŒÀ \DÒE1êHò_¹´5uF"uÉzô nQ³¥—Lr:; † _£±ÚzE–ε›—INøÁ’†2˪#4åÖc0Sºk N7ZÓºW!ízðg…ôGµœ®iëÑ-U 8»¤A•~An™ÐOo`ñ²ë\ìKÀèp! Q…ÅZª4íÆTÅöŒ%ª®K­  ­óçêú-ضsŸ¤ Š¶è‘hCE¨j"äz§U7ß Wèh:„o#4x’s ÍÃ[æÕ““T` ´( |SÚˆhÛNfäCc××1†´Ö·=ÝŽ¨d#<˜Nñj~€B®å!öË¡Ú=±k#4·-½ 1F£"::Ÿb6ůu’ˆ°Ö(§¯º^Ö(âm D|O¢(·–ëã8ÌÏEõ§‘Z uh5µû³Œ×½ïÅû‘çÙÑÏVDî…Ñgᆪ†„=æ7‡$Iü#)ÊeµúãÀëa ÉFNÎÞ^#4>Δ"oî?‡KÖdÛ#4ã°÷Až{õƒ×9æXöðp>äÄý_[Ç#4•Ã@çÑ¿,mÌ\Ém X@)¿Ck@-Úè#<S»lï›QÆÛ9Ú1åÖ•¿Tæ.ÎÓ¬ŽLÒè¤[œ}#4?}3'¶#pc~ÏL}©ÂÆÂë`!t‚ŽAG–yvSÔö_!±Ïž—¿HÄ޸b­ Mâ fÙi €C¢£Ò”dp0.£ÇÇrÎuÀ8%à'÷9ƒlò†7"ÞKYr³€É«bgz-áÛ¯*, ¡kê›T_·\É~¤aë×­ïÔ´ .G@hDËYAt†úJÙP,Z[ž—¼»¬‹Ê‹Éíõ—öx'8B‘[W¨pˆaÞƒG$8‡… qÕ:À/£w<×à"&£z•ÉpÉQ1 ÜÐ!@nŽ~hõûáeÂÙ`PfGH7tÈS…/¬c4Y¿¦þ_¯Ú¬mPìTi)¨#+ÐÂZ-t$ðÌ#<º5øÓ ï¿x”ߧg6—|–ݤ>SU#ÿ„ÇDX Ûý?^ÞoÑïö˜{¾­"â7 èo¡Ëô‡wSç!ª«J€’?è³Òüî¥_*Š$£ï²Ö{ó ÐRD**Í|¬>ߧÚ™MZÄÇ–(Òtíöuû{IÛÈùM#<B*ä`V^þÀƒñÑËKE°¾ÃÀq#)“û>JÚž.sß´rs0ð‡Í>ÎT^ÂÌ#4Õ£N'Ôe:ä®Ïø/W}½ypýƒáYþαâužXb2·(þ _¸<Àýæáú!hoRí¾|ø[ÐØ¿ä:"’páËuñ1Q&½fÂÍßàr÷Zü¶óþ/ØNY[öã»øôõÞßtfºS¬ªB ¡`÷ˆB¡ïCÔ¾Ïì€w?ùáoÒÚíúMDŸåCûB‡ô YÁ#)._ñÔ2e_þŒRáRKUP©ýÅ’1)K¨vÿY¢¾hŽÿñ´4l€G`ÉEp#4*7ûKB 3 G[BÙþæ?¼¶ÃË=ooeŒœBíÞÜÉt’ÿí›ì”h°?ßVîl?8dø&`w£˜âtÖ‹ îêï7Šš… vIÚ܃ÀëYsfÒ—†ÓUC—T!.e£× ÈWh’C®hÞúD+ª/ÀŒ#<†³fË#<Ý„¾#<"îýðÈ Æq Œ@oX¼²ÄzÇ´ÁÖ ÖåìÃÛ×L=öÿ{éṙÑ}Ÿi ˆèk†%¾ßj; ·€ŸçþÈܼ<mAwÐNá5Ô -‘Å…î\þ‹ºJhdƒ–th?•ö` >ÓBÒŒ‚v?>¶î)è#)Öz×áê6 ÿ5…DÇñ~ÅíØywÚ×SzSxßÆ@^#?»*¿T¶¾a–ì2@´þôìüjmø„æ‚Œ@&T+Ø{MÍ2ä’ÄÓR«Õì‚ù—ød"9’â¬#)«Î¢W‘Ú®wõ`Ûìú>r¿¢Ûˆ9{5-d#4›€`å@J(Ùͧp#4§í{µ GUú­Yï>!½i}!ˆþiϘH™Ê²Y*«H,]ø!púàÍÇÞ[S/•-AõÂÚaþ­]¤¥_Ö%·—¹ Ðê\ià~(pÿÜ<ÿ‘ ÜyMœÐÝ‘ò"X%.3Žx`³]#w>bЯæn%@Ôç`óÞM4Pä$C—=ɃNðµŠ.Â'Lo£T–OÜÜ¡ ô„iNî2QÇ´¯¸…DDC¯55[lÚœ¹'¯åÜ kСgê{#<aÐ qÓ¹·P&Å 2ü;™SQ¬uìFÕ÷£‘èGU£øèyš_7²ÁhÂéd%‘Œœ°P‚‘.Á:“ž^t#4–{ŒlïZRŒ¸@.=[Ëd¬AîììÄ㨳jjÎ â^£¤CDj«€†-ˆÐZ|¤?JO¯òÍîzð@Ä"²S#<ò½zD ]’}ß`^äñÿ?ˆì˜GÔVfÛxjß”3È1J‚<7†_Úä³Å›mÿm‚±oñ#)øçÄyæMÉ-Á¾?mü‡ÂûX` !•R¨‹ Ö!Á¼"•ù¨vÖ]dÕ ƒ*=J2:,sJ¿à³¿|a°Ï‚ßûs]“cÄLD#)’J}ÓåãØG}%+ _·Æ€ôʯm½Ð=¸SPÞòѳ]:§Â{OÒxA“,£#4‘¦ë &—}]PÑ!ú±¥†]qñ!ÔÁAÁåcËóöátÒô™M¡Ãð™ŸZÿs;Ë¿nÀÕl¢xŸ‘ÈDDb;M;Åõm(Üí6‚J6ñĶ`RÀÖÞKH2OpÒ{|MüÝG!†~¯ÚURÃñ<¤qü£À§qPITÂ#4DK¡ÆçCc{²u„:ÁÎ&`t)Áÿšg€js‚Ý¥s9wg~“ð¯…ÖVj#7fÁT•{òS²gkS—»¥—"Ø·F8Æ+`n*ìhŽšg5Jª£®›]šÂ³_Ÿ23›ß}àt‹ÀT`S8t-o¬Pþb|¢—¦pj3ݸò˜9n#4o@îpua±å˜^!»#(p7šÇ¨Ì#4Px»½š´fijl\Š,Ü¢—âF”õ h{0ã'Á½È€7ëöÈ‚`./(‚@ü?ÓÕõhô}§#¬…HÅý1mãap¼ "2§Ê”'äOA€<‰?s9Áã|¤ñN# 4ç´Íë}RÉgú<â”'§à¶Î;êî Œ¤ lZ( @(`á£ü~7Ùö1-*<#<`PŒWÀmdX,ì6;4†RÄ‚ŒfþI¿!’ñg|Ã-0/ÊÑØ¡i¸X1#)|¿sbQI÷’?¯/SjÓ*Rˆ²0D‰(Åã8º»ç| Mäaºû»Æã‹Ž#<ƒ|hˆ@1#4moÕ¸^ÀþqH˜›œÅ/ Å3"ˆê1@‡Ÿ{$[=_oÛþìÙöº}ÝZH4PÓ"¼ú”ÚnM?@=CS„‡˜‰Ñ “½n­aŽP ²ˆ’Ó¾Xbˆ êxïi1P #4FD ñÂÌOY˜q‘N3¥öYP~‰Jž4/.ºà>—zÞjSÜ–Ã>#G©¡¯ #Ž%Ü€z‚ÉrŠH符J`ǘ##Z¡‘fî„gwBNîþ±‡eýá¨Í‹Ñ÷ÆaPD)k­Ïðâ«ë#4]FãÚ‹È.YÓÞsì\ê@#©²Bø”\&Æ\÷:_ »ë :*#Ó@ÍY$ˆÜа`6‚j!øŸÙÕúßÇöƒþ¬ ~H2ʸ32ì®õ þ9—Ë£%¥¿eÒ˯öÜ8×OŸ:¤ß ”¾vë»%J lc`Ø?^#Ô³wý?ÌFç†KK½Fá2™¢WYY£Y§­2'¨6C´ k9 Z]`Þ:Ë2¹JD1Y7k¯$’I3Uk˜©šÖµ%¥­¶žf®jË^³™Y„uã.¡Ž½aGƒo{ÂkzDhhkµ%Ü7‡±ëXêËŨdÐKT“Ec?æ2W>¿gÙ!QŠ„„„@*< /ÛøòRÈñ#–/í°:îö MX¥ƒ~C«—•¹ÇLâÀbï…>n¥èɨô­Í:BA*ð;"{6ßñC‚¢ÅÙWOd?‚Z( çéÙëO­W؇*xp •w–‡ÚC—}¡-ˆÀÛžäú­UW$Ü#):ˆ ,„ϱè>5Ĺügñß³ûÆÍÍ®`O¸[î)_KBê™AÈÓd㻂¢Ù¡RWz»5ƒ‘_Ñ$¨m> ÃÂI$¡¶>PlY;>Ûq¬g&…’¸izZªÎÀ«àU¶#AIDF¥#<1#4#–ˆ(êðp4ïZm½CL÷ËãÓCk/À§™áœužˆ¦UJ~Í1íÅÒ/™Xùp1ó½Ø©µÆ††§œ)ýfÅ›ÏHu>ts.œöÐ%ÎÝÛö†ñ³’D‘c#i„÷_¾þÓŒÝiJåŠ ”Êb ‚ÁZ ¿M.¾ó½¾Ô½òïAˆÐ+äMb±H¾"¨Rƒ_°Bô…¾;bøý_R€„9>¯â-ïûŒVa¨#<tYJG¢äxØH]~í¹ö·6éüÌ\€ó¹BW#<P÷ -ò8žÌ9:îvûŠ¦ŠZ¨HBC^ÜÍ îªjeº“K¨Û ÛB÷f‰eBñ@00” |ÒÎt˜Ÿ£X?NEgÍ+ˆni]º‡üj­¶›·¶%Š ⪇֨î'kÞÉ=bˆBØ¢Ãd¸€H‡™ ÐZH~ö)hÈ'x7 1v,Â1<wËø…ðÉi(^A¨59Ü[móê¸ñ@£P.£÷D0ûH …Ÿ«^ì0ŠkßöÙnæP´#)!_àãïûroqö;ÌJ…òÝÄpÈ}6ì¿#)HAHjO{ØT¢ ﯮÔI'×…àש%½/`(Ùï U¢{¢¢˜m6¢yË[ô¥I‚õû®Ü˜=0}?{ý=Î×-Ò{ÑÜtŸÔ‡&Ï°l e‡O†aß9›qdPˆ0¬#)óI­ Ë±Ìãµñ7‡ß-ðê„XÆœ@ïíý/Ðì=AËak9ÅØ‘ uù€ö;6x§ã7¨sO÷Šv‘R‚$nŽà6>]¢‰0™%â+RêÝË¥wx»Œ0pv´b#4AÞjã°%hè8ï—|îĦÒTþóg#)ó¢ˆZqä7ªýq¦•E¤ÍŒ&£iÂ2}çZ7ô…ý‡0ºp(úÌJ5øê¹Üœ¿3k¯w#)3 ÓÂSä¼â-Ò¥O~žÃ’§ÀŸ×V·ÛöhQöq¿Y?yû‹¡_PÿE€_ä†ÓõŸˆ·âsî“oÆ°á0,JÛ0~|nùê>@ø}ð_GÛ^_U¬#)f™¤>’ FŸ,±U_µàêÜû5íþ]*åê k#)D€>¯?’¿•"’}½b'\qÖÔjíèBÏõÏwó:|½A‚=Êgã¨Ñá­­Œµ;üqÜÁq&`ß.©Ê¾=ã#)1¹èû¥†Ü,¦ßúO½Òz#<#4‡î ô(öAd á.Ï°ªþL{AŒUüªO¢#l”N±Ü,‹ì9×”#<npbMA¡ay–¡Çh8÷¿¬îÙ,sËñx3%û“xäÙg«Z´(ÉV-U¨ˆÓñªªLÙÈ£°˜ *þt2ƒ:’8ØÎýù8D°X“‚S/÷R¢rÙØÖ¡Q“•_w»¬†d@Äõ*ˆÆA$U¤,ã«×7Ï›¶!Œ£ak¬[‘¶A›ÍææÄïzÊ蚟VÿRlçÅÈŠd#)Þ?qãªe† ‰ ²+ |: %¢Ô¨Û×Áò‡QÚnmb%#<m±ƒ?.Ó¼ÄO”F«•Ø£eR°k—~§w4DÍÓÐäzE;Àæž&aóÉëÀ ÍÛç0Њ‡ˆi°Ói¹ì³ÿÊ–ÑÀÌ$P§`³*¥¹à{{þ'ö>Êù'èŽ~îÀ©†avY3*¡ºWè·óõ*ø4’£¤ÿ,(Ú£ÿeRR±¿ö¸Ør +¸ã`£ )$PXuÚðÅɌڕ(¢)TMVn}]}óÔwÉÝ<ö¶ã>5õΰqâ{ 9D[2i}'â„OZ‰oêú$—¤5¶˜`Qó&^çj‹ue†OqÙ¯_¾×= PìXû ö›>Csñ›:‡q 2¡ HaP˜µóSÅ8hýÙ*Ûp:ŽªS¨ø“$óÆü³“™^¯l)ùbvÝàgÒ¿mß»^_,eyÂ'€F>:¼&îœK0Fá >ó#<¥z…#<…£o¡ùLžê¶2UÆá`)ÐÓD{û§—^ŸÏ5@ÅÇ¿ƒ>G`Ñi(ö3ÊÌ ÊL{•”¶³¼#<Â4õܹ™ °ÜÕ[Ö(ÒGš1+•5XÅX%Z™2iq™y÷Ú¨¼£ºY$Aî'ðr#)õŽ… îˆ~jžKO A±±7¦/`=°Æ„ˆ„EX”U Du.ñDÌôçÅ€$Ò &ŠYÚ‡¥@ñ¼Â]ȉ#)Q„ÏQøh™Ô¢hªšÂÈN§yM)Js *@J#))c#<Ié}¨lØáù"S»q©L¦…¡•Qt  èBˆÐXƒ!ÓüǶ~3ôñ´úÎÎ*ÞïÌj¿]R$R+ÕF&³çD†<-_:6’`0†˜~rÙn¹*!F‰LÂ#4+ðïm¿r«ã\Ý#jÝÙP”A¬µ-»¯^˪ÏNÞ‰i$#<nÿÁ:‘õzüÿ~rÿXpNCÎ »îôÏo¤=ϳqtZ#<Ëèºçòfˆ¿q ׫>D3FcÓÌóFÜ’8†`Xã úÍoàh¬Ä?Xa7,AoT¤Œb¤ˆˆÐ‘À@†¢N®‘cPñAÁ–9r»H¶ÉO/'” :ë¿‹³wiè¤Ñ„„%Ï6}FûÎÜ©è/Á ~ÅB¥æ;†458:dWfÑýSÝE$åZEfþàÈꇔ#)#4g†òŽÂ'0/¯aFú![Ghrlœù ±ÌÔuúçmô·ámfXKïCˆ§2€ÜjbK#4Ø[8T#4CŸik·£ÙùlöÜ‚ùwû^Û«÷Ûeþâ|CÃâãë®E(Ê¢”Dääû@þ>ãæÀ_€´©^sÔwÙO¹õÝ>¨ÿ"‰#<Òié§Ñ±£¹½º•À†2ŠŸ×#<Ðú‰’[&\2™´Èåª/Y‚¸ &PÞ…ž…’U’IÚ!0}1²7B^t=bÛÀÐ0ÒÖ©‰ƒ2¥¡ã”Po°9þÏkÎí¦©)‹=õHª]S÷Ìùs¢—èEœ=”ub‘ÚQ±1qBùv?н‘!¬D×­HŽ*BCšjÀDbý@tì²ùŽ :º¾_hõ•%¦ÒŠÞï»X¼ƒ°Ü†HZL:¸rêÒ )Ü’Ð{SÕe TFtƒü“ñ#4ü{ÔQWô&óf4&Ì~•õut$yKxŒ+¬xèZ#'PéH¬Œb‚’Ïv!Œ ®$‹½}wFÍ#4Æ8< ´ˆ gÑ¢~?äûd/ö+³ïÕšä éË~ÙVä«À2#úãì5a÷~R°ÏT¦'ë!Oðæ{>ó88Ã×”×",¢ˆ"ƒê)Uhª=ƒ˜Æp ÈKœOVƒd_‰Î‚ÿ&ÐTソ‚ V+#<È"¤£à…”d«Ù8¬2âÊpㄵmÌU!ˆ B#ò„H§åH¨`üˆï%/©Çæ:a—Ð+·±ðëÎØòÐBÏ’!H¥,E ‹dÀ(p ü`í ½÷9†îíEŠ@Änr裩5"š‡ò}c²á®~¨]9ní5ïGŒ7BnW°>÷’˜Š‘ÀºÉp#<˜#<$ ³5n³‰’½ϳPõ)Ð:PnyÀápÛ$…”éúlƒx‰ 4u;ÃÍb—¥Í¶eÂì‘:mÄ`É—»OÏYuEY0daT Šú[äD#<žCÕÉñ¤…8ú#/o<|·/œqg{³ÒƒN#<ªŠ¨6vf^YÏÊ¿CÃN,&îºA;زC2¬‘ºõ›H|¾§ŠØ €t#)BèSÕl«((—½óûø^„‘T4ü,-‡rB„¸™“¶Øœ&®íŠ1ñžÜ•|ìàÌÕ’‹²wlqȇǧj7p|Oðk“ùˆy糈Ú0dã>›»†/@õ’*ªU'2n›š+tî¦)iô楇Jf¥¬(A©|»„o*6cvÙamhz¼ §Å#4ñ<{ÁòÊÁàpããÛÚn0Åìø$ŽÖï:WZw¶q%È¢æ’Z¶KQ¸î㬄qs©2:ÉÀ+äPw—^˜óàëé“( ­1 ϯWK^Þ‰=TüûDx Ún#)ê#) `âäO†´›ë‹ÌÐ&½™T=¥CJ°€;Ôw§3ƒq¤bB9’1âA«ÀFÑ,1@)X'#}Ê+ò„ó‡“Å1 ÝßÄ–BÏ ëÅ¡¨¡!‡{&")3CXrÞî¾}‡íÙx`G·YtaHÊÌüF—Ò¨4À)<PØêBŽY׎uÛ¹S-´]£Ýâ>61(ä”NÏ°j%s÷£[ °QK—¶ ëÄèÜe¶(êû“ëüÇð휫û@#<“P}‡ÄÇÍ鳯üE%ˆ$wÿAñ½4²£ÖGSÆñýµ…†¦²gDÌ{ƒÃ½Žs8Cʃ-ÞÝÊSœ“ÔWÖ§¡QPuw¹Ja&?©ÊË“(oÛ›+÷¹úœ§?CÎj·ŒQœž_êÇá“wgYÜ~ÍõŒà9f+|Ç«Í}6|ðÓ Üvð#4mÊ:1vÞCn•/NLF&0ÓIeñÁŒÃv…Ñâ Ê5YÇBàZÃlèb¯_/`>“»ÛîüÚ¥UsγõœˆYç#<סJ’ }™–ÉÚàh:]Kaä#):d\~øôädUî/s0ÏÅ#4H,yÁdráBA#ùû… ïà€x¦Áâ>)Š„ @ è&Z*ö‰#)¯KÐÞìå««6Áhíïý¼á7:!þ£Ûa.¾ÅÃË’G§äÿ5òLÞ"¼¨Á6r#)9¤_õâbOμÛU#<ÔèwÙ[Þ’G–ÐAlÐÖ´ªî°YìÈ~¿0ô2lj®ÀÚ«U ¡ e4#<ö}Þ¯ÓöžÿwÊ&'ôUµzT¼Y#<½À’‘F¯¶—uÛÝ»CËy»uV~ÃñÔzfô ŸÖ×#4jqb?Ï#<°…#šç&¸üÕ2ì‹žxàÕrM¼Š4ßšüè tŠvFm”Êë‚U¦ñ8˜•WæµÔ °#4è­Z´*怘FB*£}ߟ\þ>¹Œ–ç6_ µ‚ĉAúD ‡ÌT?#<Ïí÷z}‹ì‡®Wm¬>‹Ðú³ø~KB'È03=h1µÀÁ>‰p€ü“ÊX÷™¹ð8#(#)Ö8É’ü»ý×Gyæ8'#)rn!³ˆ7ªŽ}û#<„„›X’ÓÂúέë Í£M¶jÛhC¨HCiKåúoÈÿ#»ˆ÷ÛV¬µ™ED„B2Çñ;]wweggUsž~îLUŒrØôÿi ZÆáû1JöÆæ§EˆbZ ѣüzÄm§¢%#ˆÂتm±ãjÉûKpfH†4Ó“@@M ¶2O‰Ë¨ÿ¿Ï‹ï;UçÏÍP”ÐvˆSÓú°hUù¥ÇæxdMiKøMÓß=§ SÍ:dÕ¸S_JÒMDàsœÂpOkí3óÎNÿ§.XIõJ#)¥9ÛëðŸ¨sÆñ¿ªjÎn½Qí€}Å᯽Ð%zªÇº¯šù£°1:¿»¦µÉÃø¡#µïT.x˜ØîáïÔq—œjÿÏ¡ÀöHV@!!žEC*Uø{ˆ'¹ÕýZwãŠ#§ç«,ÐÆ–½Ð÷¦Ÿ­ç`Â1&¾Û8ÃNõ°¡TKôÔhÉÒjN5Á>Ý]¼jœ–»{Ší1W´Phf/?‹F¡#4¶#)Ñ€\˜#;Ø`™éÄ#< ™Axzú¢_—ôÃÄØÚ®%ˆ¦^8ntA)#)#¯Üž_+½²åÑE$4b†mÉK×#)ø…lô„(„YlÎLI“Út<€)¼-0e„\¶_mš#)Ú–È*æ·ïü—Ä—?“_SÔœ‡ÞJ>ûÍAø~JãÙÖK´Upæ‹ýNT-Ü€Çè’{Ñ‚‘7”3¾/…¦æŠ˜Ñ#4Ø‚²’Ê&“„#<Ñ3{t̶OSÌïw­¼öÂ4¢Óô•x¡&Š–È`dÈ‘F$°a™Ü ª ÔÜ~¦#<æY`8œIÇf,´±L ª‰‘„he*œÚ ´-SEðŸ°dkúüa˜v#4°†þ°ó' pT…™³$¯RÔeœé D Çqp Äñsn–îÿ.6«=kÍèªoî­fF¦Mɼ7$ÛdÙĹ:1WêtˆÐŒM„ØÎRÖ…ÃÚ›˜Gb?Üpˆ`±ˆàÞÊÏ|#4!ﺉf4›AÏ• Ñ¢ aýDOû6¡7øÝõ=vfÃO DË #4‹ÑoÔŸ³P#zÔí?VÆa’âÎdT‚§óÏñˆXv×àV&@¢¸¤ÝA“¸¬Wíj\Ìæ}-ý °ß1ííŸÎ!hÆ¢#)`øÝ~<=$?‰ý*ƒB’ÑbÂÐ-CJås8Ç, ~!q<ÍsOSŠsþâA*/p€ P¯V`¼9Eu<#4XNÚ½+ 4YÈði0â¨n†.nˤÙØQ°ØØè…]°Îj2/z‡¹žÌC^á{†³Kµ ïP{Ü‹A_c²Õ–2Ô™i!ahª,<’ ðÀ`<J#¸2;Û൯xV©ó>~ê#<oL`†p8@¼j-!Ûë‡qI2Í’BZZ±F#<~¾0`#),«ÃÃÌ #)õùäØ¡ó8(<½#40õõ*2…½Óî}l”ŠÏq›?MI¯{þ‹+Óˉ_åœÞ 1ûa˜Þs\:ñjØóëôF p”eyxâ™V ’'A¿×þ.Àe1T7Ѐ|wpÖñ-õšÍÑŒEÕ>ûèÙ7ªÛ¿_#9Œúš6HÝÚ8*P(3ÓÚÉ=Š!z‚%§¸#4Ò¡_’s\ì)(>RMï‹8Ám öÏò铆§C:÷b»:£¡žYå5âà¦ôJ&·±ê˜Šf8g:óA%ƒZÔ¢ÍìÝ#4|µÁ¬úˆÍ/ày•ƒxÛ3‡«,ŽÈQfIäõ|ÐsTp²ìôÐÅ ÐXÐŒ¡âWAÐEعìÿaè‹Hß+"îZ½¹¤¬3‹³ˆï¯èk¹;ì} ÆU Ö‹@Ÿ’wòí!y •õ„ |Ãwµ˜rJñù¥ã7õ;Öu°•#)„À‡Ç »ýŸäì1)Í5¿Ïq~¿Z¢$!#)æz  9@gòýÖîû{Û Ÿù¼ÕEés¶p•G2!ž?’áÞúï›ðÁñyEðŽtx~J½ºíE8dö||fh‹DÊÔL#<s‘eÿ#)Å¿›¬#<êÅîÖDÛk•B«/hl3èÇ€ÜÊCÖ?³õÿ¡(5‘µB¼+µ‰½è¿Wß”5‚‚wØǤLMï²@?j£ÈOಇïÃ÷&ßÞÎ3 lk´8wé»vú¶ˆè„ǹÁÇŠ:ú$øAGúíçæ~f–xÛ•›±í¸tgCÅ8¾?~,ÍBÀKeÕOåÞ|ŽB=¼ºröÞ=€hŒs>²à9Añøx|Ëj‡üŽõðeèì,3ù®XÕ#4§Ôñ=ΉïŸÒßÆAhèåz‰ Þ¢nûŠr¾ïÚ¾¾QlcáK±rã·‡Z/vóé·©Ú£ùe£/Tìr5] Í‚/„å«%Ž¯î½sunÒÓ¾PÓWߔơ‹¥¥‡TjÃ"ë…Ûv™ÀVÉ~‡Ž±îeÑp¹ìP21!7YF@ã51r#<¦ç(´ÑÔ|‹Ðpln¨çÇo=˜“n—àUƇ3ˆ¤ð­»?¾·ÇS½_,—”óˆ$ß™XMÝn€Â0ƒ„Ì:ìíœ?cxÝﲞÃBÝ<¸%K–öÍÊv½"wZ³;.Œ\cv.z@™[66.›X!¸¿ÑlJå'i|^¶›Î%/…Ì#4-ƒðUK¢ à£;U ¾e“ÜG\ãœ6|òþ=9ƒœ8›ÅÀ|éÜÄ\¦•ÕÜ—m–'”ã/¨îtËôÎS,Óf^Z(ÛŽÌm†´3U=hs[¡‚€‚Å( `°U!½ ‘€ˆ#<NU¬òNXã—gÊ1,Θw<;o Ò%ôq÷v9þ·çaÄV3ÉïýÊt-¤0á£cŠâð3:î&!tµkÔN#§ýPЀ#)‚+Ä|¿»Íïawê—¬ÖätŠ6翸‘G‘ûw¯@a–åpš;{+ש]f!ÿDÞÐËUn²+Â?*ï[ôׄÜlWéµYpÇ‚5'~'RÛkQÂÜ\iEŠ©]b\D>l$ÂK”ÂÄb=΄¼‹¥®¡ÖШÏi£0ô¼¯¶µ»%þÕô÷)×/§IÙp ´_]ö©·w•ç”HfMù$adv"D¨¸#)Ý­DH­Ás#×uÕ×®>sÊß»U!¢v0¤?RO²Ý¨²çáŽæøQÈÓ›Ó¨½5>}|º‚ý%ˆmmŒjÇô}vŸŒ†¾°ýûÀwËá©€úëû“³Â]â¨ïÃÐ7‹Ä¹ø]xÚ¶„àG5þŽ‹ï DЛ›wBŽ+ÒÁä#)•‰ë)˜ò»¾ÙÈäFc¼d¹d¿v ¹ 8Ôþ#)gýâ2„ƒ+6ö™õqºH™›ôÀç´.Bq¤#<€•,z7~åûœ7ÉÓ†'gŸXl‰éîs®=)ßåÉP–1cÀˆˆ^p¸Aî%áêä#<mÓfm²§ÛG#NibÑéqpÞçîz_­êE<cP¼âL¡ª&\ŒÞ&ê¼á¨ƒˆa®Þ||ó|6‚ù4#¯qŠ™4ÒC@qc&Ì”¯z¨x8aø°×Ód½xC´­2ùúyÞ<éøK`Ôš)ÑëÒî¼4ÝÐÚ|ÄÀÞ{Q,#4ouõÊÌJ)}W3¿&Ñ,g.ÞH‹vªDÃÈ™˜GÁÂNÞ¬ø¼åÝ-hwª¬Ÿ|á#4Mº²ez@˜Òz+‰*•Ãˆô°¬†îxŽÓBEŠ¤XFÖådº¬‘å犯GDÃ"oÖ±b¸®4Ãçœt²÷>tø´mOÔ’TRëú»hýsÛ†#c#<8–(³ãȈHȧуý~õǽà{ýœ$šä³ÀP‚„ u³ù÷ h‚ŽãïFZóÇ'2á6^ÿ†«È݇碑AßëÀÿäýù{;ÜÜàȤ%0ŸÀÃ÷þŠô™Ñd#4#IÎX£ø´Àý½Ø¿À}ùÿ·úŸ¸‡õ ¨3·¡$»/ôc‘oEUU‡ûÏCº˜ûöf0ÿY½&ÍëlÀãµÞb@9•¿‰2#4}Ü^½ƒÌÜMÍ°éw `oÖÈdÜo‰(á&¢äi¤$¤HÅŒÔòOí²ñ#4ºM掣péÜýŒf’«=“ Ù¶;ªÂ`Àò¦5žË0³²Inl-˜÷3{d– ×@Y ¹oºF;#< ¤kès;éDzP}Xá BéÀ:CV°í{Š 8‡$:À;‡…øpq#)ðÓzb‚»±|Q£Ì’žŠ!½S"SDk3Ì,̆¦VÌþóc¼ãözDJ*•Œ•JÕR¨ÕØvI¨u;2Üï3ÌÔÂP-c°ß͆¬09»aÀ*Â|žGñoó%}ÕÛüèS€lJ@0€^6„&Àès9tŠŒCË¡å2uuÞ‹|%Ëk˜<<Êh'py¿.ï¡ÆøØk†ìXšÆoŸLß“Ho<¹œ÷ÕR®QQ—™ªm7{vz9„˜`ØÃA¨ k A€±4#ˆ#<Q¨3K@E€á‚y#)Ê;FŽÆoª7ahgƒ1(Ì°®*óÈ^g¨ºxv#<Á´±EŠ(ŠéÎX„¼@Ú>ÿ«]ŽÊé«…ë)’ p‡#4¶ÓGBM{¹™Yý³Øøëô#ý õ@:}¨ßO››“e„…;µwsµ“±týþþ»Ý.DþŒ>]±7ÓîîõÔYž‹;7¨T8§¡ÅA$8Ï¥¡oéz#<4.ì$‚1‹ƒ–”ZÉ aÿnÆãBÍ«Lm“V¤¨"A¤F0aì+@†\iþ`—E C?À€”帒µ+ý&Ü™^¯ù?êêßSZ[3FcXßàßk¬/kÚT$‰¢FŠÒ•D²0~_å5§¸)_TÔgÁˆµwmÙ“Ì wá Þþ÷ù@ !÷iœ0dõ½½í`B-š¤B³—õu5·ú QëÔÕV+áÇ“´®vv–<C…ä‚Ù —¬,¹—5ÜC[‘¡¨3°…´V¬„Ž-á{×Â,¶®MBwØ”uq¡ý󗛃T äÖ µ‡B„Ô±¾42½¡Cs|“™„aÀÔ„1lÓñ!|+8ÃÃèM‡³ÈsküäRAj¯1SÝÚ9‡â4#41±¨4EýDTß°ÑÊ°ûš`ÓÕsãå}õÛ öôYÖÝùxxàfäMKÁ „ï°»¶Ý‚çäp!FøOx°ØÛð¢Ãœ’áÔ^®8m¼Œ£@ÀÏê8< mÑ›­ˆ¬ì›Ÿ¿†ó^–V¿:·Ô¥ôÚjÉd̆f¨²—!˜º®€p41x¥ø“£½ÆD¨®ÿ$]Y•Ž¾z¬ç1V0)¾­\b-÷ó¤ç<K;ÎX£Œ#</4‘×µù»døË!êÍ<õËw'’^}Ò… y2•!©4z›­o&îûîœqJ:9ézTš70²#†ìÂmÞ3¬ÂHÝŽ’ºëÔD\·Tlä¾pÆ·°øœ8hBìÑ)¶[‹‘¾ú‚;J7°¢Lœe¡Ú¸âÛÅÈ7wíè*¯wF›÷£4{qQBóQJ{1mÙêzõ Dæ`[3q¼6b#4Õàämaa3I¬ÅÔ|?ÅTÑ[;½^Ëš‰Ù¹Ñ8— fwÖd–3žX($®dzI#4:pÛ—w?_—o2BïBø)Gµc;’hhÃUÔèã»#<.GGª:ÖÃzÀ#ØáÓ¹[9¾Ø§{Ǻ(5•¨šPc3²œj6ÄQ†´i©x6°¾äŒušøƒ PÂmŠóYa¥y[P¶ =Ý»¶0ùêl“ C$µàˆ=#)°ãA#4!rqíg#)eŽDñö÷GÄ4$y.—1®…žnp¤×ˆ_-›Ybœpë+Jœ£ÛÈZƒ¨£Dô:bë9 Ö9CUöUÈÔ¿aÚóñ f&†£[Ê `"a¬#4°Ž#)wÇ*Â×í«Ëã8Îr©R¦!5;uX(ƒ¬2^ùkaÌ4 ôI0x·s t׺í+—*òñlø9/oX¹/v!˜_og]±íš?ç›9ìx°¥ÔjÀ;GÛ‘q-ÏZd\6°9[Yg5¹n¨QaAUÑÐwîÞ„f6†ÄT×5e#)ÃWòÖ¶Ô*à@©ÔöaŠ”M¦í2&ÂÅŽ*tl–v ЇhÀaÁK°E7,I¿£ã»xµ”’ÖP÷rÀô(EM¡’ƒ,ÛØYŠhfÓ—cÓÜjdÏ2Ça-(رåëà5\¦ÌÙÏ’&„¦Æ§†7–/Û*övB’E@‰ÅBìÒ#4üõÁx¾+ c#<³TÒÛÕ7iwºDò3Š1¥À{—¿ÌM wædä<ˆLÐ) ,‘*‚¥œZ*…Põ#“x¦|‰AIÙ¼ÚS/q9J.šÔh–1ÄÒäʉ°¡îŒ5½X~7Õ»ðÖ75s¹+)¢°DDi9°«U#BÒu(èzMíÃa-:¹À€Lpåíš™‚Qí#<s.<ÌÉ—332ÆÙeY–fI*ÇlÏD?+à«ŽìöAd(Uu¢¦’¹J¨F()@aLïªQdÆW®†beóë}Þ¡áMœ#4À퀽„6O(­<òËrLðiõ;Ûw¿mèòÒŒhlÔ„]»ÀmyTòþ¹z§geº8ÄT³:ð\'‰‘F3#4@Ø`ÎJâìe uÕµ$‘3!#4ZD¶´ãç'[«ÔdtÈMÙ˜aÃےΙ:rª°"—g¨/múæM]œ%‘ðÓåØ,žöcF“.륽ÜoG]¤”d· âÚW~Çט´Œ»‚mÓ"†Ý/jy€„ÈÚ ÜBÍš’[’úpÒlkfßwÎÙwYâÇa·Ôø3Ú¨U$Ôu2ÀîëÁéë#4dÓ’CÀÌG)¢”Aknçð7[¥Ò!D:MûÑQVúæ'-@…TèÉLÛ‚“ðó:dÿ*³pÔAÚ#)wnah®‚79ïàUCȃNdjédЃ–î‚á “Eˆ*x]eD³*ø$ƒFÞRÁ<ðàv|¢^&¤à}|Œ´~,¾'Íß3ºêœ¡ÓVaߺ`s‰ÀMé,˜d!üV(Õ¶ËzÔ5Öw­Ø:”„€Ž_‘“´¹Ôc ]…ò:š²hÝ/p6é! .$GnÖ9¶<²åȲŇôs¶6ܹiüƒº/'ÜS-ÛöDƒñ \q^…ÛBhi2s¸Õ]"yøí#4½Èºˆ#)1¤ù(cL&™¥eÎI#)ä>Öý²O sïn"ƒÓc¢Š‹º0é >ZÆKïp¶ e‹ÊûÊ$êuU&PF{%‰ÒÊ“!ãÀânWˆ÷¢H‹BŒ2uÈ:u:×Zç™W%°™ðÔ´@°oæÝÆ>q¸HÈâqÚÚìƒ×é!º;áCV`£LÑtˆ¯êèPÄ_‰4¤`ÇEjEQ(×­Ó8’F%K8€L£àeeX'ÔÜ6Õ3Þ¢ÝÈr†ÑS X–pÐ^Âö#Ôyü׉žÒ ¬7ò‰³'Z\9ýlJ vL6Ak¨o KT!X0‹ÐE(ꌃ÷ëí41"z»fÓ fa&$18N‚¦¬{s5+SV¨Ù˜`ZŽ¥Á"KʈäÁš3I!Ü´Öµ{†—(ÛåõŸGîÅYˆÜsl# my÷û\ÁéðÈù΋©y· k¢1ö®8”îñ¥!%{hB^ºs‰+™A§d÷dÀì RÉchšlµ2ŠMö¯´2(„Zwt<ç!ÌÄsb³s¨ €kMg¸¬'Kò)³õú«ZÁ~¼™¬Ë Á:=g­ì2£¯À Kšš¶5¯Q  ÉÔBÍŽ§xS¹Ô§€C"åEƈç~ÅrU(LE?D¾T#4ÙöµÆ}WF¾ŽÑ”‘èǯ—·di¡×ƒu+ßÃߢñ‹G…‚BÁ#<¼¼{{¼î†ÌÜÓ*#}Íã},@dmjÜqüóô.Êc'Æó{zjåÖžoHcHhc8,LÀÆ<`ÐKv/ßòMÒʲI,[†àJ‚zKkc#)›gîí82aé,^ÿñÐjVõÁ-Ž”HHb{uñ;ÊGɘMÜŒÊL$–Óêÿu`ßLMž¢l™°6'î?I™&0Ž#4MÉùÃãnzs÷Åšï´‹(Mï‹k-êðÌŒ#<€UÃ#)Àr:stƒµÀ —ꦫûùIL´8`ÊKZû_±?†ÔþøqŽM)6ÎßeØiþ7…Ô‘º3ÓáïÆìÁ°!|¨B#<¢‹ç÷øIh;cü°?Qð7™™–½iå`´#<%Ø…Æ ¶`eJ—# ·SÍH߯byÔD¢%;p!ÂUJ…=Њ°eA >Ÿqñø½ÝºÒ஽È/0sÓýd{¿Ñû=ó,ÃýŸøÎÖ¢²^Ÿ_äy@¡$"F‘>B¢*¢yÏûoéµjïáSí.¤J-™²JD”¡ü"¬£»»Ñ˜÷U!³Ð@‡‡]ŸÊÅÄæžÒ"m~´è‡Ø!€1“TÒ&&·×ukü-·Y™ DN#Nýx„¡j ¹ÆPE-A$Q#)ªÄ¾X0‚ÀŠ³¥BÑg®J£lV|¥Àû×HAz%¿fL÷öò9’Bw0 Š(Š-@+Ö/Ãð¤£ÊÊŸ´ü¨ÿ|‘nÕ(¦$) ZÈ#)x)¯Psy§ü9Y|hé`M<p“iîÉZ;HMÖÕQ7àuÁž(TJ€nd TG ô+—ü¥RÐÑä\,9bÑ©1#4d¬M¤¨%È"¿LOI #)„!ß?tHRꕇÕ=rå4Áò‹(\4GbE#<C¬ÄÉ=”†Ò0 H u#4„É$ÁöÚ<ƒ 7äáøæsœg¬;¦VÛuã¤ÅÏ¿Õ¸Ìî¬]®í+Ε‰‰r*€¤Qc‹Ê‰E4pŨDCX@8þÒÄ"Ù#4;;Íðò•‹8ül~Är5ÄFÚÍ›â/qÌyUC¸|n¤ÀÃâÄlFOÎ4‰UÙæ £h„/AH(zŒÎ°–T~Ê}hºÆŸ¼Jfayô$&à«zk½îÁ dìI_Í­³:ð cen]b6åÇŽ·¸õ¢¾yªù pO€ H™³¤ÅäÀ‚¶"…A$#)‚B REHŽÒ8*d@ú∲(aJŠ’$ˆ1 *ØŽ%Ä¡Aº@²¬TÞcÁø†ž~)'ªÜ¬n²YÊea;ebYUBÄpÒ5BIð KVd)Øj°Ö*i4˜Í`$ÄC„$±adÞ³Jõ”y+ ¨ÉWÆ/+án]I$[åݼøªå|›—ɼ×bhÝÎ× •!´e‚¯úìJ²5#4‘nD£Ü©++»GÏt»]~ONÛ¤ù/h)!DI‘QýXu ·2€Å#þøÌÙ6A®Ûoì×ìý›m[ÐÀöEÅ †┶#)ȱm§E‡Q cØÇÈ9“Ù…øýsüÙ;~â¹7v[$ÛN¡<YCIuE;üñ°è6oÝ/ÎôyíüªØ±cB1Å,ÂGÀ·ÌñÍÂù¿Y5Ó,°&Î'ê_«¯W>]'Q=Xì¾Ã‘ŠmÚCä•$…ò¹Xš›a†Ÿòí-|åºjÙþš{#4íâ¦LQSTJŒ DVD—^(Ùmz¬#"önMMÖIÍýkÓK–?Â'Aìøa¨Kw¨'¬j¢?ØZƒ]S~-°k4¾çþ û\#%œú¯«ŒÙ0õ4Ö@Ô˜H”àB˜ó®‘%&̆MuÜ> i22@#)˜BP†–ÐÏmînfBšRÉq¸L/ÐÄ@À‰ { kƒtìÒ#4éMÚáào?Ö*Hõ#<­ýV[ü>åÏn”ôá[Š\fÐÎ$&|7z<æËu'¸ü<b{ÂÃnÀî6H¾­U`¹Ã Þ!%Ñx\ë'^ëI gkkÐ4sTùaD!X=áéÐöÉßÝL‘`Á×D˜aü^ž`„óåÕTÓ&ÈïÿDHVÍÑöt³<5^Kž™,•d†ßÂ#4FhÛ;/Ý»âmŒbYAlÎwCÝB©Ññ>S÷ êùw‚QB5MSEC·³è{“¨ñh¢‚QÖ kOM&ª^¾SÌ⚆Ù6«õ*+nM›V4ÀµXT#)>ÅNaqÜ‘AˆH0‚/ ’<‰d•ñ)9ÆÅÊ©%T•d!´­tï vE4-îËÑLB‚ØW$œL•¶&ÁŸ¢þUž[Áf3$}¢h«™ªæ ¢#¿Ç~{ë0ðÏãî;c%UJ#<YPÑÓ™{‡]îi;v5—ú=óç[R_}T˱iJŠþH\=Ø!$$f±¶£UIZ²A³V4“ej)6e¦ÛXšU%¦Ö+LÕf4¨¤c#)C#)Z€±ˆ Ðà’pŸ9ÆßT©9C~ÀSÉ#4ðA‰ç‚+QEzZÞ€=°AƒªB”¡*Rú;Ó_A¹ïZòÙ»hƒ\ØhŸº\ÈPÀxhluÅQÍD†(´hÕL.Žx‚=«,Š\Ù`ëA©¶ÛZìڳݢ!§º²Ë"°DÎ0ÕAŽX¶h›x†6#<mÒ¾Wø°âìÙU•PiÆ´™A‰%úN³`±Œ‹ŠÐÇœëÜ·]è×#)A$Q#)$#){)T±$ØŠñóä¡–Æ cÆV¢ˆ…²µ¤H&ëŠdÌuÐ 2b%#2¡bbS_ôÆðöáúõ0góç@gŒMr¤b€•T¦ä^îÁ”B ·€Xᆉ|Dd$F$‘dX1ÌÕÍb4›EcZ¹·-¹[šeB–)š-™ƒQRm"€-4Š Ÿvgá<šÁ ‡fÉÚ——<J@á $„VDQÅQ;ÐPÖ!Ⱦœ>Ñûï}¾£ñü0·5XC(Z77KHfdáp‰·æªäA=G"ò¼çxªw]€çT5=ñ4#)ð™Òí>2Óòw~OÉöÿ?ßãþß÷ë5êа(yD‘ò›ñ¢Uó1 #),îõYà—ŠÂ1U•R`ném¼-¿wÝÚÙeýÛòU*Ó5ØT#4A¡}2Fp~g‹< ø|=ÔÚ]0$59´¡x ®ÏL©qú ðÜŒ~ìtȨ#<Ÿ[?¶´:« ™×L‰h¯ÒÉÇ#<殟‡’ÔzcÍD—þtG»ÞþÇîLjýT"ì™±šQ#<Š9nõœFƒÓ†Ûrð;“£n¢æ¤¡)CU8{ŸäÉí;WÓhlTfðAG·Ç7 †è#<f@C¶‘Û6#<Ë Ü,p!E10MíŒwGšuþ³§×Gh]{r$°ºà;ÁÓ !Ðg$þ‹®g¿žÜï²ïÙ0Ô¸Á ¬Â5§“©ë˜¡îÞþH ‡puGø:µ8eûaGÆ퇯yÔUÊ%NÉy×4Ò¸hø|±˜¥U 4I¬n~†Ê ‰(´aö7Âh!î>”ëå2V:'\e&1éκh½=Ù9j¢@Ï‘ÕpÐÙ©Aöló£ yŽæÁ†µýDuõA2lj³PÚât"ܸÛnRhš,BÌP¥3tB®e¥pÇMù q)qcïa—ëx²[81ýsûy[ÅnF¨“gW ƒxmƒLYT8Õε¢”pªq¯ìÍïkŽxÞ´¹'Q¢<Á‘€ÛO$ ôÀ‰‡“2¸¸æci8¡µ%首¾px€, 1!BzÚMV¥CÖ>`óQ⠚ͧ§Áø(;|á»`Â?Rо|5!nßåýÔCºD„,ÿaûýyCJ Õ”ÑTTQ¨ˆ,jY÷t<Àïæ#<2/ U#<¤R-0h€H/?OêhŽªd〉êÉuu'›¯¿PÂœr ±Û+ao2’ûÐR—f?ÅÂ4ØÜ’“ÙÒÀ€Fr#»0HhŸØ¥n;¿L4Z”’Xbê­YÇÓîKý#4{9Kl„ŽUlŸcÕ &³^`܃QB€~…¤ìXA3ŽDH @ÓX8}ö&•L5t®òËà¸äæ 0á±Î»Db 6ż4ßñƒÞ¡öÅá™CÇ‘`EÈxÈg«@ÂÙr¾‚àË—qÑÐhT´Þ.RäÃ_ÌáÉ\©#<IõºE LMB‡i~—_˜²4ÒÁÑ0Y6FM#<qf®ÙÊg]¢ü΢°âk×oÂc^¹8˜ïAåÑ„çgvwÔ¤Í^/´Á,»àm»;Y°¶›YeÂp¸@V&L«f¡·U˜U)°.iœíÃí®:G!;í<ÖU6ø3CÑÑr£Ãaºt¾¢ëÛž=Øʉl7<Ï2AËE’r –ó¨FNL^ç[:òk;PhÕÃg#4óØ+ÕÜìØÁ»ôű©·|úÅ©ÚŒLÐ(+«vn™­üõîõ3»ÍÒ¼í=äZ¯§#ÇH¢<”v—®Nfm0³8¾ˆô¾lÅ#)…¦,€Ô&7ï‹¥#)ëÓyõã€.ÅÙ š;7É#4‹ãÜütîçs.Ü”ÝÞÓ”ÛÕE„åÇK%£×Û; !ǬuÍD*{Ì/#)„¾”F}"(ReÕø/Ú¡,Ï(¥ˆYBÈð«ôÜýÈÈO;±àg¿XÝÃ"xŸy}zt×Ùc@­¡Vœ R‡€„„UèÇwç ¶ÏÏ?8*rÑ9_Æp"¢Hˆ“J€Mú‰E¬ñ­B1D~s=rÚ©Öc@_”"ÍÞ~!‡ Œ„¡Þ1W3Ñj®GMg¿ÓY@4ÌÐÜÙáNá{7lߟ X÷†Ü˜à¦(èa®M„K¿#)¶ód½™òKbסÃ_#<ôï@6;Œ[tÔT.h9K°Ö`&7sžÊ[÷9èzŽ$ƒ»8Q£„Ž»Jªi±!#)á¸áÍönÓ†pnkE"Ÿ+¯‘#šh¬–U7T'º±‰ï÷îcÔz"€¥¾ܙȂ!† qÙ3 #)Ø<êÀÊùGW’C¡$P‰Ý˜*ðáï%‰9M“—ggûu3Âõ³ÃÈ?>¦ZkÈÙö±.Ãì(ñžo?u'ø}:–E„[ŸC°zôÄ÷í¤rf¦¾²I"÷³»‹=‡(‡)a©µ4o55 hýþ>¬ˆò5Ô‘'"wÛ¥ð’§êGN&=¥d+³6;m–NôS5@&³YÁ?¯Õ¯`„FAmêé¢òÙP ¾!Ò=˜µÕ±@å™4N!Û'm+PN`r¬ç)ÓWÀ߇µëÝÛSJÒµ]UÌidhfÉZésdHwWRénä\–5YKk·ì÷žsOyªHDA¶‚¸X*2*ÆÉ"hdˆŠix<xX¶4-%±P’À+‰AçÇBÑ…#)–¢A’!!6˜œ9A`+„PcŠ! ¢HÀØnÛËñaÖŽ!ˆxä9Ã!ÚÌù±z|ÁKòÜá¾ÜüÆÙãÚ}úöÈ›­ÂSù?m×”Š˜Ò\®–M®—m¿m×WíW]Ýú½ÏIv#< R„$…ÐêP:B<kßnȼ'æ2*>×­2v«¢0(þB HMNˆ¤Lq¡ÑAd¦‹;ëPÒ÷š5ü’@lî«\=–ÖÊc#4fЬ#<ÊB™Ä¸–¥hêY+¿ð­ñJÍ{0Ü{EZhSf´<ñÎ7k4©‹üù{ )ÐÆQÀé3¦#³á½€w¯&àü3\v[ïB É ôÀèœz>ˆ9!úxϺѭs]gu»][µ®í©Ka#<)vÖíÒTš`Œ©H%Ÿ#ƒIÑ[A!‚fdˆp=´±ÔÝ/~•9iÃb#4×ß_/Ó)¨ÆÔ[6ÕÑC\Cè&Qa' ß“z©/%BÓîɯ·œj¹À\»¸ò£ð#4³[„š#4Yñiýÿ#b"¦g‰×yÜæÀ½ùÂr)Lw<gb€c2΄u~.6f¼ÖäËKL´Ö#·mæmëZÖv>@!Y#)yç¨WMÅÇ6y¶²9¯†yN·rvNÆ[O˜Ç#4Ü­ªØ"ŒÀv•6"Z0<A îcMók¶#<x4I·Z¼¨Á°BEØfku3^@¥¢dûÛQ "´gÉ UF‹û¹¸#4{‚x@¦–ÈT‹G@òl֙à «Ävp%I°"ö#<)O¯u:áFCDÞň±DŠvÑb6ÒSÓ=”4ÛåúK¾UY#^lÙTë~½zHyѶÄ9ÈãmM¾vi[ü÷l¦o¼N"“šrÉsô67Mó 0ZeÝ÷êl/8:£­ý—½¯ À´Ý…2#4T)Ö–Rç‰Âöàú›máǙ˹s„p:RÄ?O=’k}™F‡úö”7‚ñ¾²jDïŒü|ºÚ"éÄ2ýZ¶äÌ“¦#.oð KÞI$‹çF–»í›³"GåÙ·#4&Àì<¸Ð¤„„ !°×–¥PÑ0Ð-t>¼éŠFPA¡sÄ1 ŽPàˆ<#4ÂY¨ŒƒÐ€¡¯%iЗ9xT,‘8¿ÅÍîÊõ0øÑYžjÄ‚WX‘þC`†aêY.#<ÈDÞÁlI‚J¦âi‚q¶’!Ì€ÈQÀ#4Lµ7cL#4#<.•ÏR-hŽn%á—!8á7.ì¨nUB_bAãhÆy´ØŽ†h0Jƒ›a¼å­,kœ*Ài`Pm biƒpì@‰+@e!ÐÑ eÔ¥†÷[²<¿g9ÃÏÛú<BØ~¿¥aúô„¶KËRöER§35­F”,ç&nG•@3rWŽ!ûŠOŠ´PêÇׄ;$ˉvP¨%³£ÇÒgÎÆ ¨ §´6i·d^¾“·ÝÀíEÄ D5b÷ë©m ìnß°’úY/éM>,ÅÉÏmðr~ltg‚Nvp¸–°lªìSq÷âi¤0fiZ‰]µ=‰Hz%²g3Šéuimž‡¿ÍŽ>Ï}bâ¾]f£cÌüKÄ#»s&k®þÞÂ0ErêlJå¹cúzÙ鬼C…˜-&Å Jª@‘"òùûÛ ŠÈ²# ©z!Ô*DkYñô÷knw[Z‚)tÇžÏf2V.G“µÂ*âK´n¨MRª«ÒY›KŠRâ£UŽª«hŒç’Ç#<fÊc˜>½MÿÜ~mJ¦…0l¡#)`’#<CÞÖ7Z!ÈÕ#<äq&ÐG?{"Q)m2"ÝJcC0„oðSS×–æ­óÍmÅ×Õt›{ Áájé3 3'?K <JQU¨)?å#š^‹á—e°À`¸À,Å4’#$Hïyº@수ôjÔ••ZÔ­‡½§¹[#4eÜlÞ3TâËöðN± =„b­#:$#lh¿!­`ÎÓðǸ†˜–òj(wHôÃEB57ºÏ¯Hé!Ü!Iªx™+ÓŒª§Q! LÌÑa~J0FN¼äã6²ŠÖÌÕZR[,È :lˆŒ¿º¡)#4„†I…¸Ön·Xó¯¤p6¶k(ãÓŒ’5‹>x,¹$d!fÍ¢˜Ø«‘%P¥#<w…ô°#š4l€@…Œ_7å=ßÞ•aøÄløÄ»°»¡y¼^]kGÍØêbXÀá¬àžÉ†µ8“£ÆÎ&Ú(,d×KL Y¤ÒÂØ¢š4ô>ƒ1hi±NýXŒz5 [E"¤# ÕŽ‡á奱õ®*Œá…÷8¯Pž­m‡“3XRŽ ¯^Å f £T=ËÍ–ÉFNÔ'PÚÆÔÈÔ„ô佯“J±¨­–6B˜ƒ€Æ…›†Üä3~E=‘FYÍ#4Y†SGK*µZqEŒIßvŒ#4È6æò€z°ÆSµT#<54Oë–ùk]õ­±ÈzÃO³†GÆûÙ†=| «»t(”Þª*T¼#4>œPQq…†°Ä9¯³q¾¤€Õ×­Å*TIqãÈñˤv¾ˆÅ^Ù1­ö½aw†vG;‰ S»ððŒf"ÆÕêÏùÉ6#4S#:c»r7s;èY d8‚ Œ#4¤PÐFãDUTB"Â&.H@CòpÕó´¦"¦úeÉ]£’ÔW‚Š#4Ào‚pѳ¢ý%‡Þ#lGa¡m£€õ0«o$:b¦ûS`DË)¡)5£sA”“Q)†ì4I I&£!²M›gLLb<P£“I àãm.[¼Rs;hÎ å­™+I¸B!*¢•a“ ´2!L’ÌÒUKCt4x04BhÈ‚ ,47j݇GH„tü¢¿ÅÏñ î|"ùÆêêT™žT¡×Mb#<K.E勉QŒ1HÆîb÷z¸*$tgM#<^ÑðFÙêFâ#4¸Ï'6Ö#[±p6„ÿàÃC'ÈÐáæ‹’IK¬œpä.Q©‚PHl†Ânl)#4Kœ#4?Z¡›vÄ46¶\6jfNÌRi…7Ç\²Pr;‹p 9ÚG0 µ¢!Ò(èD$QÛã@†&ÏÁb/#4¦À²:âq‰xÞ e5úk"#<r½àá´H1°›ÈËg7»±Êêà7ý{ð,pF6‰œBN“a=®ôò“"k•8É•nÎDt1êsØÙ۷ßœ¼3Ma+LÁ) ‘÷6Ö#<ʨÛ0‚VÅUY‚‰·é³b}Å!æ5ùKn.X-ŸN‰>øËœÀÜ1µÃÆ^¤æ´k›«~O¾Ô°æ–õÂlKaÝÁ3L¶É°}í™3»¹AŠ¥9½²b ›òê¡›ïÀ)ÜæÜA6‡du ç‰ÝïÜnÏ¡ŽV;9 6 @)€‰Ǩãö{¼G¬'·s‰Ã .'q*"ÄX#<6`­†-„Ì*.VÉ|–#)i2~–,V `;ªɹ£k70p)QÄi5…Éd’9‰#TІ" ú¬˜LŠGÔßE±€fŠ¢g—4Ølë€Å¬Dc†*F›  ˆ½Í#4Á’&£FNQCv¾N#’}`@ ¼1ÖÏ#øêÈzñ#4¾Íå ;‰7ayýé šŒ„\ß?jôÅ“÷#4NÍÍFs€™“#µkËei߆o$÷ÉÃ#4êàÍ“v„¥ðš{ÚíV4` 0Ý}ê¡ÎKâyª#‹ñ¸Œ´S3/NÓ!Ž’/Å.he81úŒ–f\VV-¼æôÇ›±øû¢ ñPD E¹Û:ì!º3ÏOP‰óŒ’P©k_¹sËnˆ,#4¨ÖÔm¯5\­U¶´j¿Öj«ËmæÛZ¢¢¬ë­B„¡vOŽf°¼<Öî'¢Œë"!‹XQ%øéFR“Ípv_l#)ÖÖ]”„þHÔ|»Ï Üð©¡ú´YB'¨XÅTLB ”¤ÉLÆj”³2VRšb“j6“-'öº’˜miJ˜6h‰B‹i£_na’6¤Å’Q)JÙ¡L˜™¦d ¦4hEQ¦&”M/ÉÝ@FÅ‘¡ALi%%–&A™4j©¨¢ŒL"Œ1ŒdÖ”lª*BS¤¨4ÊlJŒÄh)•E“6D¥#‰ŒÚàå«<Hr¼±¦õæRÂ…@bl¾p |_•¦+©v‰y7}™ìÝ}?Ž²´ŠùJ}c8C8£µG0«½}fÆuv@»yn“%ÈÞér'á°4%¤/b@Æ2ضã0Ø—`FՑ˨Ð;7#w›E#<ÊcûèøÆêrä ÈÎÒÞt8,ä#){ŒpÔ¦µ¶}T—f´Ë#4XÂKߎY}FXia“#<·Ü×±7ðÉÃR`¿ N&Ÿ7µ$ãE }ø‹˜ã²@m&Û e+x5[#xap`‚.’ß±ú„ fÄŠ“X‘6ÐišÉ¶-Œ#1ETDHžÎ:öu®ùƒÀ:‹¸8Qo#<¥Ã@WºSOÁ¦±3í-Ô,ª›ÍÎ0Ã× ³_…Õ¦%ÐS8»#45wlÈÇì­ù7Ê^Ç`*óz|Ä€ö´Ê‘®Ç”öP#,&Q]ºÏ7R|ö/Öêž9^aŒ"o‚=Ò /¹‚}µC9‘lŸÓï³`ÀŠ›S<Ó¤p£rÅ%ŒD]˜Ÿú Ó7J!ºª²Úik#T¤!B€º#<HÅ8iáÒcòñüð#<z9M+Ùnó”Šéç6tŠ—x4ûŸ}UWDÄq¹mÅ>#4ï˼8õ-v¬ËáöZÖ“´®>Çý°ËƒÐÄ#&gL³93T8Lyf[M»S‹÷½ŒÁ‡ÁÚÆ ÍúÁM#<‹jñZ¹\œA‹ÛK±‚îBYi{Y6…æîÛ–8ÎÛìE/>%Èßl°ü›í$,7ÕÆé‡m ‹ “ic³4X¸=7œëo§Áùœ>í‡81j CÂ:þÌïGžS«ëîõ^^¡¹ê\—¶FÉÏP<©Š2‡©ni§q=敹¯º4#4²maw5É5„Ûa? ¡,;b“—O¹èóªrÞR,½¹I×¹Ï>27%¢žÈïŸL#<öeSÐ.g'Âí™Üm;DD©„ó…s.ꂃ3ŒfÎ68ërŠ"•‰JüîŒå4$–½­ˆÝ¶ÜÌ‹Þ준‰±÷¼ì@ïã2ÄTý>©¥âé¯IèÛ¿Á³Ç#]$P!ËoÀe[é©Ô'²»»ÁÕ×céê,xKüÞUÁÎùlN„)&Á¤ œ°mT“"2“‚ €qô0Q+9\±ƒ`ºï<¼T t¢-G'–ª+À<4·íKîj`h#<4gJA9jTµ24W¿¤ïCH8c=³qÓΙ¢ÖÏ9R\;;<¯ê‹¨ao]#Sã%£xTeªX]5”={ ¨ÝC?J“áéÛUMUŒ2ŸÝÕ ,#4 ÝÁISGvQŒ%œóÑ2"j3ÉI¸ŽÄêtÉì~÷RlMÌ—]¤9|4èŠé;?x@‹&P$DàVõJÕžtŒ=·ÿ_Ÿ…Ž{ó2Ï_äyªŒ>uXû±Œô« hͲ”“íhŽF(ñ­HÇÊË™—f£fMTAê0y\(ÈÖ¸Ó£{šÌÓöã3ò¹÷ÆÌ1¶æ£tæcæi5#4$&²;ºE§Ó’ª\„hD‘/iŠÖ o¶c'–R<8s1Ú²io5¬Dâ­dZ.Œ7™3#9ÖýÙ&ð¦ÕÕ“P¼ímó§k¼o'†\Â-)oöæ´a.syÑ¡o­L3«êœp)º\,@øuCTI¬3>š÷ºD䂉™¨*qi=ªŒ\Ì¡2Ò¨ŒËµR„1)¥!É"âî(ˈÔD£Bâñ;SZÙ8ƒ 8â?GIÒ^åîM<ahÉwšÞóµöÙ®NýÌBÐ>Lƒ¦q ¶Ä“±GXƤyÔÇHÕ£µdÏ\Ü×N Ë‘»l#GWv¸»0¤åÞQb¥ø1•¦â6•}]¡¶}s6'HE;mcþ<àqÒÚÁ¸Yׇk¬iÍÔIF$wz`ƒc ¡»0#<?ftÕÝ*ÇV I|Ÿ‚CÉ®¯M#²767¶ŒL±±‹!ôC­¨iÉQñÇ}æías/úË Áñ¸!9Îvånª¹Œm¨€x¥Ò¦VŸ¦c-nÆ“3J|´ÜlQbÖî^0ÈLºZ97ã2׊—üÜ9‰ˆ!øv;æ#<LŠŠ— Ä;P›e“e)Ô£38j¥Š ÉQ„&ÂMþ¶#<òî óÿhß²1zãPN¼"Å;tFQyÄcÆ2齘z"V®ñ¾5ŠÞLëXÜÎý'ZÏûìî=%$mN"UØÈ»çxøÔ.î¼GÐ\kÎcŽ}s¼ëx‘tèbo£mÄÇ'››pQ ûâ$é>4%&8s5¼©ÎøÞAó°ã‚IߦתÞù®N͵£x Lи—Þꥯ„òã‘[åñ®s©n8Âr³Ô%C¶rNŸaûnng;#4“½±›Ú£ ó<ªÓÞÍQ%aIP¥ÔiÀoÄÀ¸Ó;1¢:Þc¬!X<ÆÜ® y§YM†dÒ!ÁcêuCÆR È<-«ÈnvY6#4ˆz gv„ÚfÒ¦Õ"hq±-kŒ“$ñËDF¹iu† ŒAA•?qbEaÓP}"HuÅ&qT¹‡ªHÂIÔ#utþ•NL#4+".íó‰#<ÛWVK™ls¡öw/hmLI¤B&Y•‚æ²FÆÁP£~0L׎¬´'‘öî<'1t²k{ÍiWÖ¨ÑÈò•2æ#4kv¥„Æà˜3¾È%jMt‡CÍRn&#<ÙëZ$"©“&爉[ê«g’ÇH¶I¢MQZ4ða¸D-ÄY‚_hÙn˜¥Á¿X-l´t✖0‡%5ëÝœæ2f€ÖÊ6šïœ_–’÷ètÜAÁôµ:xç¥P9Ðo7j$­h„‹4Íkk*’kˆ]Œ×X¦¢mxáA9ÝÎ!ôÙÞ·yÁ›;Aè/Rc/S<˜re•®Ú†Å0²Ù¥D‹-¶‹Éó·#4=aÊËåÍ WLB)U Q´*#4ê›J:ô"¢\…{ÄÊl©‘UÏ#4Muù¾!ÆÞ‡©Iõ¦¹…˜q™Ð0µlåm°í¶/<¸…p<ÊÆ]ÀA„@ W†wÌmfu&ÿ7£|p÷¡Ú^7l&“.œý9.2Ýô™tjFÛ­pó Ü#5&©_ðñ£ rl‹y_¹²\îâ>ÆåœzØkpk”çºO·W5mÖi¾ÜÞFos;:Ö`6qap›2v“Fä}ðËæ°ã5£3É™%3½<­SãDí@šéêtT­t³Èp4CŽù¥³e³o$¤©æÇu­Üc|ì&쎆o[¶fœÙ™YªÞªˆyD¢;D‹z;Ï5#<j°RðŽxŒÞ4¦9Ì¿Rh·:TLÔ]USÈM*‡‰1Å_‡[¢¸ˆÙ òO±1 epSÀ¼±x”Kµâ×*Õª…ˆ´Ò[A£ÑÃ1TLÆÐ9¶jÉ«ˆÉYÄ#ÆàܺÔÏqšÐcÁ“&™»FŽ.:jn¬5䇧‡á&¨6‰½2Íffjti$ÞknmÖfMÛ53—N½Fcelvj˜1<“’‰ÓD¹§MvÖrÈÛ2rÄW^ße5ÚpÄØŽoãeìœ[Nï£9YÙªŸRŒòl6[Y;B7¨û¡‚×i7ŒÚ'Cå6iæ·œ"k*Q'±ËN+X­èËV(‚q\p¤éßä$‚1Ã8²˜pr9àâÕÄ#4êÁü Ç8Q£˜!9…RéÞ-KÜÀäLTfÁ#‡"„liŠ6„ÔÍNT…2C":`6+<°«É¤Éq$Š$}íŒÍJüZ–5Z):U ãŽg“zhÉ*БãpV1Š‡M‹-UM0öæ„Ç $Ƶeѧ¢NéµkiìÆvÏ~!F¦¨Û×lÊrîÑá#)B òÒøÕ#)ÒKT*Hž#4­GTlM› ´ãs†Ž‚… ¶’œiI+F5ezÎ~WȲú¶¹à…¡mÎV7-aa–629'€äÑcl™•‘¶Rbi@(R `nƒe׸E45´=‘uL3ÓB‚ç“e"8£ÅìpÄæ(àÑÇØ †ìïÕ6®ÏZcªD-EV†!.ø„y‘wÀéØbØ[$¾•#<AHj“tšiRšfY2Í4cJŒÖ¢9dÏŽ±‹mh9`qç#²L¡¨‡&ðe¦­2cºƒFnÎ#)…²h› ”-†RvQRh4ÄÚ¡-’™Â©Hº&{+FgJ,G5)#4Ü0·£I CÂnšºf¦Ì”Èab ñÖ÷g$–‘mZpÒ¦é«Å4‘&¬œÙ·œS &¬¼Ô¦™4@ËF7 \>É´˜¤šCÛ1ÑìãÒlïXŒrý8Dt“JmÚ”¥Ð¥åfŽH#<Fæ\0™·@Æ»°Î˜ßf#4õ"e'PHèÊ ß:zr¹—/\tŠc1-˜›e#)éKÓ‡”m‘,ðÝÒéÃÔî×’cht'9CÃî»ÎÝ m¬FÑÊm¸(ZF™¸Ú¯8Ù˜Ö”næ9–èss‚rl¤!fÈÎh†FquJ—ŠQ±ôru‹²×'2b¼KÛOƒ)“jˆâ¶“g#Œk&/Âã®×&˜bÇ„7#)‰5T^SBgj`E¸cc½«œÖ^SÁ¹Ê”æ2µäˆÆ[ÝŠÀÒb&Í"ƒ$Pkl­ º5@Ï'6÷¾ü3d×C’òa“[Âtºˆ‰2Ð)¾É!à¦`€Ë`êC–ûtF8%†ÒÜa(Ê2‚4&ÃLŒÁV€Á ,¸`£Œb±#[x¼ã‹b ôh Mx<CŒj>»3±ÑO Tlïä2e‘Œmµ$`1³„¼x¼~­s †\¶:ñÖ5Œí6ëÉÄÿ¦©Ï"ð·j=Ž´Œ.}°('•éºh¨†Aê61Í`Ðn= K†LÉF—&,Ö à9M{Ò¬Š¥„8Æ™’#<ÜÀ™$ÔS=´àFMQn›áa8 @â¸Rˆ‚!È9Ê,#4X‰™T“3pæíÉ€Ø#4÷nHcÉŒ£C¦#Ã<ÂeU"7ìr@"„€‰i¼ØéFwãÕœPQI’0\Ì´†‘»¤9¤Ë”Î2\fðãÆÊÜdA#)I ¡,S²l3›#E6ÌLø˦±É7Õ6è.Çc@¸ý[ÑÙŠÐfÜÈt06.&h9¦¼DÅÔ¶À¶¶D±¦†Ë k¶R9kcŒ%Á†,Õä’ˆKºÓm#4ö¢µ³ƒ¹°ÒeKBGFž*J¬aá G¢§d?‚ÐŽgR¦¬Iš2Ym36µÍ™!#)VÒ“÷E!\5½™^@»A4(G¡ä”Š0QÿCˆ/Õ¼=¼œáBG@ííV•$‘$‘ª¨Êª„€$CȪ71þcÈ`“*°Â#<n#)ä9·H§ó#Êo-ZßzŽË{µÖö®Æ$Æf€#4#)`Rp@raGœžS—5æSH&HXäv„±'ÆŽb¡µn¥¢r¡;H>ýí|®H|ý3ƒõNël}IÔ'yÙîeáÅ´În#4fR™­c&ïÚ[H,‹JV“.b©mMiÙᤸn-Œå) O˜ƒ–V>NÕ[öBR>/s pòI¸›fõ2Ma¨3.âä´„qÄr´6´àÂ12 Åpv èu,:áòˆx„r„8åtì;‹¿_B@Ž½pë}ÏO£üÎKʆt%ÞêB$‚PGh;Ï÷p¹~'ãõjµ¯-U³n“§7“Ôr¹-Ș…óhhHLhÀ1 ¢)h#<¥ÌÕ†ÊȲ‰¿Æ ®Ãu@âo…|òBÚ&T†=\_é!3È6½4jBJµµ‹5 ȬNeñ¡B%ËÃ*ɺÞÏf¯#)3$;Å£®aÓߊõªü‚¨€l÷ìR$@~ #þXÈI"Q4Ú$M25.›µ®l•º^Ô½»²5làT´Ðêï ïxu"Y(Y͸ŽÙLv(.ðp&ð8Ÿ#)HC.G<ytÚAy@å¸2ˆZ‘10ÀŠ{Om~µv”g˜ú¬j? dÄ»øÏàd AØâýM•f¿÷{ùœIN[W"Æ ¾z锎Õ®š´W Ú¥Rb `0(Ô` eŠ#<c´M¤ˆÞ'ÐneÀ¸=¾PǸ-þ­›)Ni1Ø#<i‰@ŽGÁ W#4‚F«QeFŒ'1R.EÇ 1‚Rwæ\1¢Ã¬)*#4+Ø~&!’òå_—à(´Ø) ÞÃ*佂q>³ãFßuCÉ ¨ÊMl[m£cQ‰ChÍ3-‚£[H2%¢fTBƒ2Ô…D÷®ïQ‡†ªÔ#Úy·Z›6_«½i1 Wytt;`)¬ÕÚ!èYÔTi£I>WµˆB­!¯d”É뮾5UÕtÔk•Ê &iqŠ58A5ô¬C'ñt1;‘¤í‡¬ e–ÆOZe*UÏš.“î>íL‰‹Š\!’GÂëz4óÍyƒÁ‹4}ké…n®ÒMØfæí(9’b¬¦PÉù]ˆË2Ëy'±A:²ÔŠð( #4†ìù#<ÈÌ-®²<ˆ…Â#4d1ZWp‘çQö0÷œT*ÀcKi"È£ÔBn_ÕÍ®–ß‘·É|jø"ÔÈU¤®ÉŒ#4™JY°CM"´qŠ´¸fh„[‘ƒbÓ‘Ï™H™±Ž ⻕¬CœßDÚ–ŒûH†Ze“`5}ÝÛ{º×#<”¨ªýÔ·ÇŒÆê>a˜òNþ3]ß-QV#<£ª7´Ñn;Í!ÐÂBac}mV6E ijHTŽ­R—#ÖˆVº¤nC€Ì%†S „£¾‰5uI“z×d¸"^lE¿©#4o5AtÇ‘Ä\b;Ø"Ù#)XD&‰'MN+ñ¼ç…±ñü~`î@<“ƤcÝäSr‹²H+ÆV‚R#<¦ @ÄE€°Tþ@é¼KèRŸ$ µ•Íš¼~]‘#)2UA‚@C3’Ë1Ç¯/4–N3/Ki¾°ÌfŒ»×#4‡6±ÒßÛ˜ÑFÚ4ÏÛ¸ƒ…A­õW##<·7³‚¶ˆÈÂd\}øQœ=vÉŸ‰JiÉa <H¶}-ß`6̤!j¼¥<‰ÈI­¨B)»RoT(0Êw×PÙ†€·dSÇ KJÚÍEØf¿§¸aœ22õÁ¯UèÌíÙ`Á'‘NÇ6q•ºu™…U‰ Q*‰(k·‚Prõ!.âpZß Ý—s•€[b!Óï¾ ÒØ·Â  ¨þ­' Óìëdg)&9»(˜wg9Î&3- Ó9œaa"ô¬Mu¦—‚ †ö#§³P6¼¥}°‰R)”mtXQ¦Ô:sŒ¸,»W ¼ÓMõ|ÖøѺ‘Æ g>qââ¬{!S­æ‘ggjޘЄ ³Ç;#4ƒ›–z!¡Ò¨aRi Šk«­†A˜À ÜÑK-#Àn„"dL‘ fÈ K’ÉÁ L LEߤŒ #<¡$PRä Öƒ²›Lƒ`c€#4ðÒe¶š#42ЋŒ°?Ð#) "!r;@HŽý{BÀkôÊ}m}„€¯êÌ™ø+”ÓŠE'²ƒ#4Gø«t¢`×à0üe¬>Ñÿ_TŸªa —:Úúlk#4¶¡Ï÷¶(&¢$?òc5×jbi%öG¢û‘}t %çNîæîvi¿=kÍÎ[AmFÆÔZÉ´jÖ-¶ T–Ù!$BD„‡¡ðS$>Š}W’¥RÙV¨ƒ?3.ße”ûH}Äï±8£øÀèÒ»Ñ[SÇטš¨ß=,¥¥÷ï•þ#)Ý&°VâÚ;+jâ:†«I+‘CdÞúéÃYÞ%)ž„Q¥…s0ä>“©qK-µ)ÖÙ{ͶɌ¡´‚™õ7cRײ”bŠºRˆšI¶a¦† Æ!mlªñ4á‘pñRmá%…ÌÞI†qƒÃ#2™¦'wl{²#4VIQŠÊÆÛÞ¯’7•Šiȉ3ÌðÍ™ ÛœTÉËëdA_MoD‰œxœómˆöèŶo,ò–Æe(g ê2•¬ÓGŒ¬MW>fÎxe§xÓð9¾K#<åpôܺ6דr1¨/ ò¨BÕRc7VÍPàZÛ¤¶v*ÕÚvÕšÍNûÌwHñ“|áéc߇hË3³ê3+5‰Â®I&öG.¹D¨ž'z·‘i’MEZ€Êé[¬àí|‡Ó%ÚVI6XòH;Š|ªè}©‘†É±e†¡Ì0_©ŸÉœ¦#4HPeŒS3Hk$F,7äc‰VÍÒ”a*S#) Z‹LÁGˆlYF¡òÛçq‰iVRÓ e%0âE‚7ŠÕC²L,Á˜ð0i#uîŠ5óÕym£zËjÌŠØÔr…(†!$#)/ (ÙP[d·¥TOÊ©{Zd/¹Ô°s]uvIUkßmÎMÝÖºmMwh»-Ò£bîêø“Ÿ=yxåd­Œ‘¸AvAÑÙUmƒLÄì;[½Æ¦šón†¯{[­ëf77dßw|î±]ÚæfÓ´ö뽚½‘¹\”+u+r{×±¼®Ä$RDŒ6e?ßž*DŽÑM£mÉ_[qþ[ko’Ñ”­3lÉ#<R‰²±¶-eI”¥—«­tÕŠm2kK Ù¶Ôüúüvü¿>hµA‚Ʀ%ÚKjµBTj iI³\·³Ñç+7æ#©GæFæ˜J4‰q/a¶*ÉlâƒÔ@‰h3yÁ§)´Q[|Ít×Um«õµZÄìNá #4ð_<GrD='yì°z²ÂyÈF3ý,(vü¸b]LÃ3Nãg×’'Žµ’È„‘$ât ±JŸX'Öl i33—„âÿ=30 gÌM(.„„HÒ6þºzŒZJ.#<4|œ›ç}õrC­å†§dn¡æ©5˜R§û_ÂÊÔŽéRª„h‚HPl¤BÏ㤴Ø"¬Ê#)¨ÊÄ‚4—nLjNýÈôñåceð×´¡ÕF9­·CV­V-÷dæ@±¶Ÿ¢#4ôµn²$MÓ#4ß#4ñù9Œl…H=+bÖ¦S1ñÞ‰‰«ÑÖH(Dˆ6Û¿OWíxÛMܵ¦Ç{ ÜVò±È2àqÕ‘|ÂÁV,ºötÍÐØ~F2s>’|po}xu(<d\ˆ«7.8ÞªV¿–›)äôÁcyªƒV$Í[Lgh[)C±h"dý̲!+íŒîϹB\Ðþ#D#4ÝrNW$3™È—+‚ˆXĆ¨Ú¿ÙêÁD ÂpsT' ;áýš’×–§a?º%ZuЩ6uÉÇ·Ø@ýpëX#)(@ì|Þ¯¨…užãîèko àobŒJÉöæYüQ0×;|ÿÂlÀƒðMµNý?6Öß|vMhÖÿuEY$Þ릚3i>º×DHJMæ涙a5$ÖI‹Yêmk’Íe±”¶“a¤ÒO9M3H÷n–i“M´Å,­¤D¿k]aHÕe3#4–Í#LK6Š-jj©¢|qFÒ5R—.›P—æÚæºãY«ë®ÆŠ˜„ ™š«15_'R­&¶"-,ceW÷6Û¶¿dM_;¶I²(µ‹"kjªD“M³k[—m-L¶úWI©¶ß>wŠh›M¦Ê)…KXTZ–ͤßmªV£¨ónŠÍ›{«¯w&RÖ™#4#4_ãy¯m5I¾#<íš¼ÚëJ%V5^&¹1¥­¦øøA\þõ/ íŽr‡ùV‡‹Ù7^{(?Óþ^?‘\b#®"óÔWËåÛ^Ž½\4ƒ3ª£UU¯Ë?…Å.I0Š}óLZˆÑÈÈgçðé~,La}¹´Ó#4Z#<‚îJI”]“¬´#4.î´bŠ#„Õhï΂„&èm¾õ-´%>îéš–›J™¤#)A#)… îã¾f z 0$(¶i[4Õ¾ÍZºZØÀåÄКTD`Ò#¼‚(4"MFÛvÊ®îØØ¢ÒX¶2¡P7Å0Š­É„ŠÈ:”S:v(é´¥ÍH*k+Q@  m–͘–b†1‘“Úý*è¡(Ô¤Õ)­m6Í™cm&Ú†”ÑJ”Ûñ[”0À@™j+*‰hĤ¥iI#<,6›e)dMLÉ1°ÌÆXØ*##<Q)­šŠRbM&D´ %F²Šh²–ScT¥±)¤Ê”¦DÙ±‹DcL–M%I)ŠJ¤1d‹R‰µY´Ò¤(II‹&Ra4É4ÉRËTÛÕ«"b´RÄÍ©2H²¶Ô³Y2hÒ™-)¶Ë5µ&µkï­k¸ÛT›JÍZk%–Èi-÷A´U±,@¨#)QaD¨T…ÖG†%”*`ÅF-­~V·5J[µ¶‘4$X#<jŠ€TDÄ’éÞO8dÜ/†Eèr²S!×@-gfJHùñ.ùÏ: ËŽÃqÓ3¨¯Ž˜oïg()»¥××q—üz‰µ˜G¾ÎƒýI‘#)#4-ÛžòÕC€V]:X<ðµ…åzª$j†Ê›j¯Q$!ðÄ1ñ¿x*ÌvŠî‚wùa‚yC|uǪYêÜAÉdb¿gƒwc][:ë»Í›k(&cÎc=WJ¦üŽJQ»ICÃEJWi‰v_[5z7¦ø?ÞÀZcåQÚv]ºÐƒ ëô2$ fl¸Óha!¼†/@)Ìæk #M1Ø×)P¢¢¨‚ (ÁW^°Ø°âéI_ÄP[ÎRܨKƒÞ¨ùõ\ÛG9 k‚qâ¨H#<•ú»»gÕqDðÏ`œ¦2бڅ¤UÓçPä’Øþ|ýFþdêDÆ&ÌŠâ€|Y¦a!"ñ×ÕXÅÃ2üHæ‡û˜¤Ò¨l6Š{AÙS ‘„ŠPLE°Ë€ˆf¡lzsm‚JÇC†Y;`”–,Å@ «x»œ¥œ˜‘q#<zyT|ðAÏHç*^“9#<“Lª#<‹­Cè©r0xU+C¤ëö6Ñ+![8›öêræY$gfõ[ëš LÖGT5^UU5…¡¢z7`UyØÛ×Z8™)®aõq{¶RAÆÎíCVÔu¤†èmõ„AÑgÞq¾#4™ÚŠ†FE­Cèw”­•ÚA #)îH!úy~gÛæÛë´à"ƤÀ@õÜ#)#)-¶-ª1Y'TT”Šài\´˜Bd:j‹wÑ!#)ª(HRE±']ƒ™²F=”<1Ž#/ÇÀ´î wëY(¸!“ß~N´,ŠÀ€².ýF¡ ¹(Œ’ˆDº´pìÞbÂù¥ËÛZÞü>£‰¯o.¦ Õð37AÔ%?Zöaí‘@O>aÀÁ7x½Ïbèoêó±¼åëlyWVÎúS=ƒ j^„Tåcio¯•ï¶'IÛMØe²è¶ÓÿƒGªïn÷jþ7´ÁŠ9£sÖâê#üËQ¦C#)‘dA‹ØP%¢Û!c?ƒN¬¶ÍZ]Ô¤U ¬²$F CœEDÐ4Öž0¤-[`V#C¨fI3U¸LQ0#4IöïžÛãyh’ôŒêœhþî!‰­cŠ´ª‘MœUV6pãPbm9]ªB3›†D6&iVW™J<¦ØÕ{a˜â$ #4‰´®Û¤“KV‡šEÓteS8¨¥!ÆF:¾q-°cCž 'ßÆ=–°÷ä7höÉ#)Y÷@¾ó5«ë–n|u È!²>nÛT‘h†µj¢fÊ´kf§¯'©raud &þ¤<ò?¯ßÉŒ”»Ty´ž#)õ ‘  Gª#4£œ¨úlk«‰å<ç<IÄÍú²•f´ ,ˆÚ-AXÜÙNÜwf»Emë)Ýê½’$=àìú§Ì0öÀ( c±Tn6.nìÛµÂûùüþÐ>­EdcP‚j‰ô¾ŸÏ§âýui‹L\^)§·à’s¦°¯%mÆ¥ˆª†–³%äqɹ‘¡a!3¯Š„¤¹¡²n”h@|ÑÇRgHRfe€Ú.¸…’Ú’Í‘P••2 B0Vý~47‘þ\›ãCóû^¸}=pïd<AÝÓæå"{og®+ÚV¥ïëÅÀûÞ»N 'ë©4W`Ðä>˴žßÍR8`<€ƒã†#4aã°û÷ŸRiÆ»Xj#<¶ÚΣŠü¦IÅ>íýíÝþ:Ïe3 0±€å%l;QØ€¶´i…À0c! pUe€()©IIèÑôj"²„Ô…Ž@£Hy€1êó€·6}ÌÙ×¹öfl °í¤#4´¦?uJƃ4bC!”‰žià=G"×ãa‘Pi»R¦Fè3Ûß Ö©Y(î§g΋£#<Iøe׸ÆÕÃQ\•zõ­ Äs¥x©Øw_¾* ‰Úzq ;2¡UKÑ” ÅâÔÄ©ÑIS#4v·¦þ…”† P‚i™ÁEèQŽâ…X¦Ò‰¡ÇIŠ!èÐfbÊ»‘¿_q“ÕéÝ܇¼ðû»3”÷-Ñ¡ËõOW%ø'q¡XÍOÌZP„0PdÇòOºT¦Úñe.\‹éõ;Î翈°m˜þpœâïµ40´:ñÕf¢15øSsDÁ¶úÑ!´GN#)ˆÙ󈪥‰¬6#4;}Ù[qFçQt±“€Xlëb–Ö9-ÃI­Tl ˜– éŒ ‹]†cÕáp6„l6åVl‘[)«qÍÓ,ÖÌÜM¡¸2|Ðêþ5$ãÑUˆ«éE*ò$õ—¨°#Ó¼áÝfò¶gr¡q§yÔ“>çÒŒ‡ §Pê (¤NàX¥‚*”Y 6HQD*øžó’˜SèSΕ4>㹉q"{µXÁTp#)qæ¦QÍ8Cyp¿žaŒz€]ÛY‘›Ô´7 Ó™°¤,E$`HA‘æÀ¥Uâ‚+»Yb È##bÌ”$²»Æ‚ ¶&ÅÖZYN>Zwk¹—Îìg8ûWs1½Ýsêë0e¬à¦7š§×¾æ¤Æó`¡U`‚}Ìv_Ëö½yQoÃ[¹~>xL0ci¡¶»B+å®*&8ç]Ý{Ý…+Ís›ŒÃ&ë»—G·NÖé,Òç½ÖƒÉãÞöwð(™aMÀa 6 ”’#4‘œ5¬†"8˜›mzÈß2‘±I£¤À¡ ‚K`^Ç·»‘wñ!*äK¬ïÓ]áÊKQ€y9€²B*C·E].á–Ñ;{m²z²å»>ºÕ°(1ÞØ”l B£®8ìÈØKã}ÃÆVÙ‘ÉÂ90­ ¤6ÈE¯ƒf\Ø°‘ÑÚ%Ï׬;;©ÓÑ€ðG þXæ8ðôêíªïˆzï:½Xá=AØ•âöŠl·{NxÈCh=ø"ova“4¥ûÓê½Ò` ,É=Ý´‹–£Ò¦ÎÑ™EÃô¢‘-¼3‹&á‘T—TÁVFSµ…ofNo#AbÈÀ§ºªQ˜™”ÙÆ$ˆ‹¼‰$Ü[i/6×ÙË×.òêó³›~–õ"e2C Þõ{å©«¤•â©•RJ#)È¡°#)‘UµÉiEß]Õç\ë;·2\±ruØëé+Ý55oi»^·-ïT±b€‹#)šôçZ˸$C£º w—Sonx#)k\5æñ–³gÍx ƒqèwhirO§ó’eK[ZNd8ÊãBÄŽ­,FÄL%±V.O°¬Š«1’z%¤ßl”0®c«F¬›Õdfö! AÖŽwLš ÀÜ*¼Ú+}•4Ʊ2ßݪÕæ€Í³M&šÒºè¹¿"½/)‰,‹F¨Ö¿™µsdQ@‹9¤ ‚ÈS)‘$…À¨#/îýõfˆ©€ª^”wÿ6,p- QÏá«€A3ROÕŠŽhY”ýXÒä2ëµ™ÀL¤-’,% RTB„–õ10m­DŒ‡Þ…é vÃu© ,go2ËHDŒ/šÞŠ¬ì`†,C±IŠ†‘˜¡h~>ìvFWm#4 ôÜÉ`€Ø<#4¦fç]ë‚{d¢¡©ÿLt@¥(o8CWh®I[ãs[~ký:‘A³—ï…þǸ,̪§²Š…],#)Ýì •l2Èãj!³c€çãè¦ÓLþ=•#<D*]ÒhÚdj4›F¨Ú;[,PMa·ÛºWœvèb0×Lݳ ci®J¼Q­ö.a˜Ð˜øIiUµ‰ŒŽ™÷‰ „±½ìŠ2" ‡bA&!…*?Ë úïñŸW'Øäáam×¢‹ Ühh0‹ #4„ÌÝM¥îõIìÛÅ(Q<¼×û¿êÁÁEÚçdÖ´0Ps÷|˜` Ñˆ#)˜t‰eA.z¢X;-’¸›û,øÛ¾Ys º‰~µÕT#Õ#4aÌ„ŠåLÈ$€/ÕÌW³Mj‰ß#<•=ú®135Èn½ÌMn5û²<á÷霶{ûKýÿß…Õ󕯠¡Y¡ß}Š)À3Ý’›˜ýnva&ƒ‚ŽÐC«ïeyÁ U¯¨Èµ?Ãçù¡ì()¤ÛŽÚÛ,WŽ…È\´+Vç‹Þ õ"mØB. ñŒQô`žÆE¿G8_‰\—#)êDÌ„#ô ÒÅÐÚ\!MQMˆ>'Æ!ôô¤"þó½píô"yÈ›ý:ÛkÃX«_Ë*¬\ÚåU®•U¶JæµÍª±kKï¶Þ-‹0/¨ "‚Ád%#)*Rgd=wÜToA˜Æ/óA^êÒé´©]+ÚK#<p2»>ã^g ‰‰ÚsNgÂǃ֧æü6`ÅÛ'Ö<’ M‰æOZQÐ8;ÌÅág°Oíw'hÃŒU) R0•R†¨$:߉d@ß•NãD}Ì–©-…"º#4/2ƒÜXíøîù^»Å*ŒLV~®~·ÏnÒN'Ë!ApÒ³ú\o5`´JOCÔuh+gë߬:'†ÍßH‘=çBÖÓ®òß9‡:${ÃN8öã¼éë£ã݇¨d€dóV,D °Dà’Ϩáøs@ƒDý¡"Ž±yª•ýTe‡µ™I¦ƒüf5CG1ƒU ±QT*‚™ ÿf-Å4T2”2,… ^Ij¶€b£:`²þ½@F0ÛXB‡, ¤v£Oø*cì¹nÊ… ‘‰1¡gV%ã k2½Â(ëW[¡Ã$Ê*Ń­ï€ÂÚHhÀÆ%‰˜B6†É#<ÎYQ¸1 Rw`a„v 0 ”Y(¦¥V"zèš½ïo/vß'Åñ$¾yw•¹Wsußnøå‹Nê®m¢‚†YLVJÖ¤”„bÂD\æÁVpÒ20ÄÐß ¦ˆ \ÀÛ™­ÍsV-¸Q¯¦ò¤Öå¼³im¤Ã‘¨ÑÎW,{`…ƒ`MYi4r¢M+S-`¶RËRÁÕ.‰©czxa¶EXVÖŠÑboUYVÝÖAÈhïÍÁקwNèæ¹k—6Ñ´®èÑQhƹn[˜“îï.4=uÕ_zÜÉo>}Ý·Øi7ÂݧR †cV ÌbŽ0Æ‘ZNŒìEßþ#4fe„7ÉpÁê<e 0òƒ!\hPÙÕUL%ÓI@ 8PÚÏíÈ`Ãɪnž™ZM§!Ô@L£r`ãY^·e®Éü>ÓЛ Ô9‘P1qüfÊ >t¶×'š¤ÀBæi¢~(puÀûCÈò‹DgÙ”# –©+1¼ê«½×i²=vÍD¡0K£ãžÛIL G:CÄm¿£¦6Þpáìyüû75N;PèäKÍaTÖ÷È?I÷A¢aS´ÁÈAŒq­”²km-+Rm5·ãÿM_&ÚÓ¸Ôn^ÛymÙ©i²˜±¦ÕêÔcê¹9Ù.[¤¶0جî§uÒ­®¥cZUË6mË*¤´Úm6úÚˆ)ÁZÂ#¤n ,‚±jTT!?dÞýh0b8–€š‚#)ŠBÉ„HA ìÜo>çŸ"Àj×Xø¸ÕØãáöxöû¼q´õŽ078ý¾s5Å Ï>¤ÂP¤€­@´´ÿÊ©al@ÆÉAb H£'J,P……S ‰H®º¾JÉñ”jxsnyD‚?Ž.>¹µ‰õÇ(·zªˆH’àÅç®·mòmrÜ¿#z¼Ýdµ’6“nt¦TŠm\Ûš¯+~vµë/Žˆ¢Ù5¬¤5QÓX±·UÝQkãkêï—¢_:Ýš¹µ~#4VókÙÊ4bÌJI*Ñ«RM6’úûuªû#@hi!ê$QAøDdyÇAôîãæÆ!ÔÑô‘kopГ0YÜ©rÿ h$R0ŠËY#*QÁ /¦d#)ÂAÉL f#4ˆ@†P±‡ahH_DYD*ªª½ò®Q8ªÐŽ„€»*}Ê=„cƒL-ÚÚäÒð8Ì®Âüƒ³ #4àWd%ŽK‘I#4¤qؾ٠ŽàØ©qólÛæÔy·oÇÈàT#<å´"i¨—œz(„#)¯Bl*ªEH,÷l·f´ 9˜ùë„$˸ŽKëP´î„L!´Ûg,.Ä.Xr,ãË*ëOª/dT¨‚g_ÂâŽQG]±“|³ŒMÎW1¥ÆKXnãÓZ®°;:ï2exKNuvË°ÂFšÈ´¥1­F¢Ú$Û5ö•ÕF•¬Ól¿~—}Jê6‚13Ea¡ ­0/ì¹Àë7´íåßdÂ]®Ø±½¶æù5J*ÎDB÷9âŸ):õº«¾í¨¤/ØG•ÈýhhÁv$Å9–ù•»ú°5NÉ™n]ÝéË e‘‡ ȆrÌãuÏâµyL¼óÆ®ù ›ŠpÆÁ°ë­æÓdÃ9I11¡”¢Ja G{Š Åchi¼”dÙÓ;«Tî408ZHÊFÛGw©dr4Yb$!˜Ž#ãé`ïòDO$ó½ý °CÌUU‚.ðž ý Á4šOxà¹mÆPèõ:x GR† *=â–8Š"ÙàˆÅ‡ 1š0-TÝ–ÂÔZj–À¦oö?.8ÑÂI³£ Ïâ½ ðu%\›9Ò/ñl€hдŽìI´}^¸q®­r1„‘#<ä:Äb5)¢&ó®äÒ×MµÈÑ#<2ˆýì¨ÅM’l†‚,É ’€b( „#4±’’åMtß(Ê©Nõ^à’¿“œ¿[>åWíî8?VÄÈÂœGa ±à‡#<}2ÆÝ#TiÿX&@~½tÁFu¹QþŽßå¹¢»MŠ@å6Œ<c‹LŽ:²Ìöec*“ꆦN¾¡>¤ô*¥ö&¬×oÐ1Pù|TÞ‘¨ïÚÙpòHq£F²dUÖR#4©†,Íšf*¨Ú[lµ4Ø“X(Ìb”×óÊìBAc3i#4¥ŸØÚܪ2V-™•M"ÑBÊj’kRŠ¦F,UM¶Ê²Ñµš™m•*¦„µŠ”ER´Ê&ÍVhšj–‹[Q­ii³í[ò_¡CåAŸµú?2ÉÇu6$vU²Ô‰€«€!á’©×ãü‰D"#4AYmÔåm´klmu5­rÛ³=Å^£¿ÛêkãmæÞ´±!8 °zŽþ®¢p@²â5#)©Èµ¹±c'Rþk_.«ê­~³A?×ü…èq_VÊMÔT\†‘HÆša‚¨ö!H–}R `÷Ì“ã‰2BÖ 0ŠB{üü—GlŽÍÅÛDokêCÇM<SûÃ>)@J 5#)OgÊ€;”#4G±9+°?‘ˆßß¾/ó&é´‰WN%¿Sœ-'•¶¢ÆÛyvˆmétÕòî‹Ì#)¨2 ^Ô¶H`A©ëBbùmx‹ÆQä;§·†¬÷i¸­·5㉠¦4b@†>Ìdíu Mˆù °òô¨ZN(6Ò™ËèKà„66‰cÒæ“;ÖFF‘¾8þž°ALãV a`f]B×"¼ùÞÞ³ãy¼#<¦$Fkäñ_§•½îWÇœTŸ\& 8CÖJŽAÜqº»»Q]Q1ÜŸ‚sDC ví;•†oò#)1l¨þ¦t%æzɤvaÒo©A3§ç£Hó8«rE-d`Æ‘É@ð~´tÀáŸÔÑ« éZÆÔE`„ÐÅ$bDè!Ëù ÓVjqjŠ(³ˆr0cjó<??‰°õ^CÕ±â‡áù‹®1aÇ:Ëé«*ÐÈÔ"铳ý]þš"ƒ£ŸRb!‹é[¸œq”N£ùbKõmIÁÓÆ1ùò:*²#<"* ÀŸ pþ;qåžWoäù¯Fž•$!=àGœÖVüŽíù:>yžÅʳ‹i·í9zØX¹¾42jªhL^õ¯Ra¹våÒòºívÅ_/á||M1×m­Ø>´­Ù‹R%PeEŒA"Ecb‘aà”^j{õÆ0$ûö³Z;½ {Ò¬D¶ýýû¡¥›5ˆ>-ÉÝë¯U²’¥,Í¿W]E¯»®å²B4Ó#< Õ$m¦ã>__çP³ Œxà01Z”ƒ–ˆ!wDY$RFãÓÆ‘YZ@44 ÆÂñ û-@Ò»Š1´&µiñ¸·“±ï¹Ä0ѳ&?P*Йt ¤ xûUE#)[ÿU¨Y‘H®9R^ˆC®ÔÏÚPƒ¤p€˜mµŒÙ•4H‘0¡õA\ð(@ñä#·:ÐE#²^’£ÏÝœþ:<s/T–á4JqÛ:ÄÙÖóxÐÚœ™‚mQ¢80ŒÃ¢æ ñLÕzŽ ^1ÜöèfÇ–ó–˜Òg0¦%§ì´÷‰êé D#)H¦,w…ʲ¬hŠÆë¥ÚêôªM¯jj^µvjªA§ƒ`³HR§Šef‡Ïæð¥ã. ×­QÔ:À0?‰J]}ô;{|õTH tBå1IEÉE³«½f“×aõ§æL; µÙRŒÍ6j#40 FL³ÏÏF[©NAõ*=ÈÃ6;6Í;MH§<Í Ð›q¡|‡p´ÂEJ’ÇÛ®ŠÉ²Uñ®^»µ™^w²à4†ÙS='–'ć1„èÈS˜7TÎØѱWl„€$‹"2)Õ`³.âB—<é+×)’»Þö~„ÆÄòDÛe`„=Áªw(²(|šâDC ª‚€È‡YÐÆ]Új m3­˜ y '/KG š»”VA¨_á0Áèˆâš(mÒ6Å8Ѥçk™ÍxʨöysLìe v6û…ȱËT'ð¯ƒLAñÅß²*ãÐ@ÃÇlÐÀ‰o‹«Ë°p#)™aÓÞ–N3zÕÊÒOšA"GÒn†ÛhÏÎÏñ3јû‚ Õ6Ø>aÅlOÓ[f)…V‰xѸ¾Æ[®EUkŠàtÉ a"°J×T¼E#<B’ÔÆŠ¡ÒÊê1ï#¤y–Qûä¬eaF|Ù55åêÏ×'Õ¢$»3ÊD´9ïöüyƒB<õåÔª(™YÁÑ#4Sª6‡Ì**ì^OF"6[ÄHùJ<;Úæ€hŽ.´‚cŠÕ©kç}›Z€14¹Ãº¦Š£7ýe]ÑAI¢á°¡5rËIì¹£5\l^¥…Q[! ìš—Ç/“J°œö•E¸Ã Ö³¦¿ÛüiæÙ‡ÏãíÛ#:Jœ#4p}(ªÆF%U<ÌÝÝZ”Š6ªù›™u4»^me‚¦ä¹d²)#)ˆ Fa[åÞPuê¦AOY#)/wq¸‚!·‡lëúÄ6s.æ¦'w€ñ ñx˜þí îîÉ#<#»Bœ3 JÓ«W¯,æNÞG3îlT18ï4ìô˜‡8€²²0ˆ Å8æÑÖ‡ŸXH&Ì'M{Ï>¦Å`(®¬l8L]ê@\8ƒªhl~1M]}~}jå¯iÝžDÕ°ð>ü¢k B/¡özZM¬#<6– xq÷lÖž@P#yJ"wYÔÅ@¤¢^:øedKi™1W•ïtÝ{Ûy{µÖ%eRªKU¤Õe¤w:‹iÖÛzíué¢é) P#4‚Ÿ‡Ëi´ÜQAlˆV_ßqÔ<z´1{1-¤#<¤€‚€ü‰I„ ¿˜/ªÿ·Î`-FA&%šAŸ­ ¶ÃI)˜e"€ˆ 3T1>™2O¡8ßÓÚÜ—{o;MdÊ\-2á˜Ö0⸣f‚41…L©•¢Š®mëÎÕÒßzù]ÚûeËÊ#<`L¹dÖál3Te#4Y)Â|ïˆÙÛ’Ìû­#©uu|믒YÕX%10léçSj“{¦,#<Î’g#Z"9 VSÍ>i´ÚÛ#·x°ÓÄ6ñã<fÙYB@’6j^„ 8aˆ_ß›N"DDj#4!o,6Ӏȱn 61 Òìˆ>f!“dlÅ0D"Y¹F­«ëˆõp6U"vróº–Ba5ÙJ#°ªF@\2#)ËðSṴ̈̀Û* ·ùƒˆåîôRÒÅÖ(õzDr†¿låãÓ/@ÜDÙ?ѽßÝfø‚ƒ4àêq‰]¤:ÁÕBRP $€Èæ·ëm{+4–Tk%lL´lZMmF©K[÷­~æ6¤¥~½ëÕ©ý¯6‹y\Û•ÃS"@‚ú>ÌG½"0ºpÚŸ“xÆ[GH<×,¥7u6©Ç趢Š#)CÑÛk5d*>é¨8§³_.ŽGLpÄ#)1(<`W¨Ÿ ª|“Óýi Ã{’liƒº•Œ‰ó©¶µÚÝו╬Ӷګ ØLS᡿ʺ4»[4<f8e…œ¶ÍØ{Öøìf·L$›{±¸ oÙ“À&»ê·Ç€®xùΛå¾ö d=†SÀè¶#<DMDÈN¬^Ra{l.5·.×Õ¬d&’BdÉ·ÎgÓ®Àæ1`X£;æ&½œLsÕI©Š@XpË9´7m¹as‰%îcaÃñÉkžÔßÁç° [E2Ü–‚<oÌ¥ò—rß]0Ùz|O"£z…úùkÀã{m»¹ÚŽ<¼ j» "íäq­¿ãK&fG£Á“š$éß4w_*NÚÇÒçγDûõسëawœ=˜ãkJ(ÆÔÖ{Žÿ/câ)¯X æ<TEöàQžìbͧ¯ÛÏìÁ=ÑyA£T|AÔͱ b4Ús44°†+Ò€MÅ×ÇcQï3#4;kî‹õ&£ÀóˆÄÓ_TœÙDß,Íí±MFÂêèx2³‘4tϱ"Éò¡YM]·x¼=ý”Z1LGO»„¾›žW%³œf‚8ž=ÌŸhÈ=‡À(5ü44A˜„û ¥µÑ ͯüÚg Îé^îÝ/pÏ{æ¦z4ƒôº»Ƙ}Žaá§;Ô‚ÖAT{Þñ¬j+®íñVæ‰6½øý ÕÕŠapDXI ÊI-ç)h rC8nñÎ÷ê(ÂÝ[µ¡¸1 Kx;$ž²'øAºÊôð¹hª’Ä ,cì6ï"™Q“ 12¼²Œ± ƒël '`8).’˜ŠB#)-1¼(Š)ñHÄS ™0–3hw#4*Bga¡ó½r.x}|NXó=)Ä|8:³yˆ;¿Gîº^wU3 Xµ¶+\åÓ‚/ø1÷pL|ÀêùïÛÓ¦@Ó¸Ù¹ðÃÄõÑÛé×í[CN7u&m­E… Œä#É#<Lµ0dN•Ñ³×ƒt0ôàZÇ¢š:iÂ7qr<…U¬ÚÔ–òüð7ÄÔG¥>he,…$zÑÃÚ‡qÚÓ‘¬#)–/ë†À‘%ëÕ€£´HäfkæMÏ®piˆQMCTj© ÉÌ¡Z’M´FâAÑ"]ºÍ³MiM­;v«©6Uªfµ]VÆŠæeåË×v•½­oÂl£diRÕR¶ÕŠÖ»~x7;“€=õO/"²À̳ÕðJVšö ³9Z±šnÅ¡ãÁ¶ â#<mÏÕÐD›ãÛ|FRn<s©ôi>ŒÜAµ×évSRê–ß;„¡1úrr“ljÅ1ÄåµI¿7Ø9süØ ±ÑZ5ÛüU§Þ#£+HÖ¢F†p“d˜FYÀ{r#4Îè¹}Êf ‡r!2ÐVy õÿ7A¾ð| ¨Þ(e1ÖðåmÄÙ‘F&ÌÁÆôèh4#47¾&Ô YX W¸AÞr›½õØlÐ|#)‹üUçp>{Uõ¸z}ˆXÈ»!ÏÙbãBµÕ¤ò¼@@ G²XÅÜ»#4Eص[Ù»NwáÿÛšªÄH¬Ë@<åNÔ)?:κÆÌ\ÉÛø²¶Ž@„V¸fpv¼xÖõ”@fE ÄŸl7|߇ïÿ/óþMümáý)jsª5î¿4ÅöÝ'zkÛã›Á7ÆÖ°Ø@ôd>ïJ…Ò”¦å2ÙWVî#4$¡!©ª÷’ ÎAR¬ÖV #4¬}˜ó1ë?UÁëP¸B¼‘¶ôÛIE`¡I±>ïo"â‘Ó#4ê.ˆ.þÌ8‹ÔÏÑÂâ”áÔm)Üõ^f–ÆÓ­`û@nx¡’ÇíYt'¡;”Ó¦Øá²'Ý“69lÔ@~=ÓþmÈ`ñŽöÜÞÇe?§îïU»•I$—“6BÀWƒÃ4f[o½FSˆeyoÎX2&.Úá5³#[ªEvã ”¾A‡°S§‚#)ØFBçRÄ:x¼lã´ؽ¶†ñ?#Œ;þ î^-¶û:c(˜Ò¡æ] ˆôÀ²R`QŸÍ•Yt’^ˆ[ ~¸ä Û¿y5 ¾søç4³ÓfMèQ€ ñLŒ²<Ÿé¬ÝYŒu®Ó~™L38#<èjwÈ]CF 8—z±HjˆH=~Ëhˆ‰!)‘B½Ð>cE_ð=o@ [å~Ù é±,#‚ŒÆAŽ‘ÆîjzDETî{À[]Ú0Hm AÂ+Å`Z4}°à‚‰ŽÏL[źa[0™38W^t䫉WÃeç€u ‡ˆP,¿†ãƒ•ùý5¾\3ôa#4üDˆÄH!#<E4ƒÉìÞ¦(_.­.œb„Œ Œ=Öêkc[§èT^m¬ËEj"­±ªÅ¨ÚŠÛQ£k2¢ÖKlm“Q[ÛV’’ÉPœ"hØ€g¹ê|ÿŽû³ ì#44ª¦F"Dƒù’0U] ‰À R¤dÓÙáÄèyËæB}©–z°r!€j%¯ÙC´ýÅÇ«Çà`Á€)5£<Ú€ ‡Øq0Ø`‰«L_nhV:ÿ<öúÜ ú©x]5t<‚#!»xí‚Óm‚ CEli î°2Ú=±íi›ÄqƧ˜}‡”!³Èôe}gò#<ƒhy•¶^lMy‡¡6)#)¥ ° ¶£Ëˇ:ŽÞwN°#<Óß×b4˜wñ›³_ÝRj–‡R§:ckƦ‘¢‡àp9w碗Èó:‡¦‡±ÊÜT@;↠y˜¬Í Æ¢7@ °Œ'Êt’Ó1Œ\Œâ$Êl·@>g'Á˜)…‰¨äÂO©Ã!Y%Wô-U-Ù¢z,{»jõÞÍ÷r/H™eBQªš‰œ Z<`¿T-- ’¤.¶(žp`ipƒHŠ1]ND” ¨'ú"ÊÖh” Ÿô$ÃÑW¾C‚0½*HP: Ç‘œCxbFH!³*¤EV¨TÙùý}ìú²|¾žÇöØÇ5äaˆk·WSŠîw]Ûå¾ú#<c&Öû’H¡=ié^ÌãòÅØâ­–ä£ó=43Øæ*Èî,'+äjˆØV·s•0–HI™#<™T¹rœ™ä€Ù¡¨P­¹½we×ÞšO#41,Z d/ƒõXkŽˆ‚ªŽ x;Uc¢ÅÓ?“OBê ßåé@ž^Àê„Ú#)˜q~FÍüŒ!Dôžµ¸:ý:Ë©ËÎu9âz2/ð•-]±´eÕ#<¨¬Ö,Õ!®+WDµd)¤…U#<#@‡Ú›¹°´ÙÁéL#4f\h1éF÷! ¤Æð¤#<šla›Òu†¢¢„r41¦!HD0IVQ›ECE£\]ÔkM#)ècW?!ñ™l¬BÊo ÀZhA&2L`Â(ÆDJ0'“éW}tíºã­Ô¥.qÓZ¶‰Ç¨P„”kE©¡T‘º.ê5!V’™ó¢†ˆEw„qßÇpÉꙢ~ç™jƒbþL›IuwfIh­[ðšòŠ¹_–WL‹6Mµ&×:IŒöîÒnR¤Úk¾{ÓØù*ï·_<Ñ*˜+æk³A뮥õÕ½Mj"±&˜4ÛfdÆ„+1ªfWíw»]æ»iii™l–©MCB™FÆ5I!¥+,„–U3"¶SRùÕÛ×jä·s¹ÃnL›ñnÞºæ±sGeö›–ö÷z&#<–¡‘¹oËó¾Ï¦l353P¦6REµt•M¤40%#lFh 66jµº\DJ‹<&ÍZ°Ð""±T0`7Žâš±Œšcy‘S‹ªkoRlÖ¢­60qý\À«9ˆi†ÃºªÖ0‘`·&[ jm2åÄ“áÇ„ƒm„Ï•G6›D8âÎ)~\4°È1œe6†ûéFÚ†4Ürgé‹×Ï4®Ö£Z jùµÓQ¢F¾ææuÚ»Çzjû›ËôŒœz¤ºD6ÚoÖ7IEŒ¤qY²‰0Á@ÃÄZÏ÷`™.>Õ4 `æŠé¤TÒ±4Â1¶›hÙÉkdNÔ­†cªÂ#<«Ñ²Ò|ã›^÷¯rëª_ Þž²2°Q š"J´‘ã"s[iŒm&ÀnÈ&41™’[Zc ßl¹ÇŒ#O#ňjÙ”’ƈˆƒÇâ4šÀÜÁ×Rµ,!14¤bndÁ¼øÚ66µ!’ÚÈôÏÆÄi’l(A‹§‹àâZu›¡MÙ74¥‡œyS5ºš±·üknosÍQS嵤±¬açé1‹'¥3¼ãO•‰dƒÃT››[F8fc/hF—#4$nÁ¦#<á´ƒYŒ[f7·ªbU6žzB¹ABi„‰Z¹˜¥âÆØoàoî5C?Ì}Œ¤ú½K.Å:#)ÜGŒýËä˜a Øý{›6žÎ8Ø$¥Àåa~†<Éñ{tóøc¥¤ÞŽq&òEFcmÙË#<îB¾˜«ŒÇ Qu“ CÌAm)“„Ù­%Sj³|eÒì=ÓŽ.BbS|^7©ĘvθE£fÖUk1‘U¶'Jò§£ HF [†â#ÏÛÅòŠköXa¡#<aCd7Zéü¸yX@ÅßTê0@Ü#4œÀÑäHÄØ›Iî#)â Ùn2ƒ2ªÝ¶Á½ mMÅó”~zopÃhShA…(Š‚g1† `QQ¶›}´÷sµsh«¦ÖUë«…ƒ-…Ä°Èì(!x’”lF¢00„ňc a#<6¢Ú](„’ÂSŠaý-ÍÕÈëì=OgŸ\p <,û6–eœRtNô<ÏóD–ýðQ ‹¿Ëmî7ìü¢¨Lï¼û`¨ÐR9„ü>rôÎŒ³^úðÝbk]ZQ‚›'ìK ¥Ést¢¯éýWiæë å^¬MHŸâúô‡£zÅ Ÿ G‹^ý­XÚoÄÖæ¥%+Ý‘îèË.C—§2ãçŽ9š9ͼãF->Jò­f¶Ùší ÂYVd¼ë›ñXhŒÂ¼tvŸÌ¾gÄe Ráe÷ˆ«_ç\!€J-—¤ˆI!ÉÌQÍè­SóÔW¢¿#4o=Mós)„Eï;Ž4òDˆÑ‘‡0"Œ'¬æèúèmUPÅ„0ózH³6"'€ˆ„“¦tÛÆpÜ'M°’1Y ÏSµË´VB¯Ê#<“¨¬ÄÝ^!™C[râÇFÇz÷¾Ì(„ä^ï¾t%Òheþù ¼™ Œ>óˆig#ÆVÞä1·Î¦såÉbrá‚à=Ù¨BÆåË~n@y2cÛÑûµÄ*c6Âi•ª÷%„CÜøˆøË~Û熙Hë«ßdþß¿¹á€Þkt¦ÔJ;C3J4i%bZN ØÊtL!òˆ2"Œ"ŸÍÏÂ;ï¡>:“àñ ‘H"#),ªjüµýJÅnj× ­ FFÂЗ²ùD5¥… üédFYG|LQ.UU–ÝéÜ]»¦žöôõµçJåWe­Âªé­^tµyîè±[½×7M]ªºmݺd7.ÕæÔSÆi,EH¬ŒT©nqÝ·wm¢Ôš¥©¯½µ^my“[I¹µ»n½íæ5Š-Ql€¤˜!ïpá±õ²êŠõêi~H-¢££hŽH‰/Ëê}Md[BQ#4 óVšÖ]jÊÛe6ÕÍ2]¤)ˆ!±eFÅyØH$!J7B  I:²äMˆQ#<4 ¬ ED¡Š×këkÄËÙ_«åjö½ByæÝX¸DÉ’P#<!B1(L" EEjRIF²‹þ¯t¶#4Ù¢µ ™¥+ƨ¨Öe5¢Ø¶ÄšÈhÑ3ZM’)š“+M"´Ò³@lÆjÔÕîÕtoÑÅ®Ù'öõ«ø‡n %þ™,ØIÂB6JÖ¢¥ìA¤Ú£hMHh¶½·õߧòüþ§ù¾k÷y<Þ zózìXþûÛÆågi`Ì"ÑnôÖg m–«ZÛ"^&P3·Îç4Tƒ"ž16M6´”Ê4mö9¿ó¯Ý+‹4kWJ¯Ø­·#4V,Ÿ¿ŠìAj4ß2»&Ýn–»´W]­b5¶d4š£[Íï[©´¥6Öš¶WÎÝšÊѨÑm®ÚîÙºþïnoú´ÍváN$bLr#)ùxIä “ÄC š!0db©$#ÑM‹€ˆê]›qCŠa$ †¦2#4ôªÚuòÌ<íõ¥„q ÚWI„5"$„50,ócÜXaŠp’Äv~ç#)û™;Š{BU9rEæßi  âð„¼u=‚íR¢0u¢C*1B}3MÀûϺL‰ÿ]¸†mÁƒ{TɆ27?;,q±ŒetÌ“MB`ƒ#<a!rÚ•º8$Ó É2†\›};aÆ©mwŒ8¶ëXdS%ÏÝÆù>ï]r1®ÝJÑØ&5lc&+'MŸ9­ýÉŽ0J#<wuRyhÙdüJé.ÆWÑÞñ¼hÀðšÅ’óa™1%äO28Ùß%µ$çÅ0ãÌBvs™ª-6݈!Bsº´@„ ó¸kLL¢‘âQFòS.L4‰w{½ïÐX*ÿ);¨V1"“ÌX#<N°ÁÛÛŽíÊÌË÷;Gça;w©»ÆËCù±´þ<¼¬åÕgŠ)“›µ¤0lTÊTŒd |ÜÑç¯UyÑû:BSØxÁ>q„D„DY5¢Ô–߆ô¶µÒ²b™oùyŸâ04Å@FBKP$±4 $H(² 騩hŠZ Ûål0VÎÛÕÊÕø+þÎ!ý(¸Ï®‘$#)C$ñÝs ¤F~„Åüœc¹„2"À,M“m5×Yöï·u‹Vå\̪½ }èX`@ zwÖîÌ2 [XÈ[T‘.!§SbÐÚŠE ÀÑ?‚lÒðÂà¤\€[²¹DwÁY!U™&1²jÖ½¬›ßY*½ÝWlSéw•u‘$šËšõë®múò<d£E#)M4„Áˆ]†-k7p¬xi:²dB™ƒl mkV¥)V‘õaV™•Ã#<!DQ#<#<ŠÆ,@Á|ö!ˆé€4™¤+ÀêÃûä"Í ! rÖÌIѤçH×Ü¿.$'¸„XÄA#y7•$<ßçâëì• ª‰JBô³àè‡ýÇï’}ì;[A…‚©ÏÀúµÎš»xÈÉÅ—råÀäýFdæ 4cçBZ›3ønËnhŽ³TxDD e¨ÅUQD„ÁÄ „È$6þŽ³Ã<¾ÓæF0ªôã£+Pby³ÿoë^†5Ì£t?;©ªQùF\CcJ™þ2bò9ÌŸ¹Óz*”ªaéUt•­¿VœOÆ^–5#„äùñ<­]k>NÍû³’ñž$ÛgnM°ÝÀÿ]ÂÙrñ€ÅElëG)IÑ{¸Ø!¡»íS´8Ôuž³p#4÷<­Rš"C]!#<‰%¡ó:šÜQ=Födäo» A9{‰5s<êÎ8u¸`Ý–„<©#<Ö6ÒLCZ¢«x`ÒPmg©ÞQaÀ;{D2e!‘ªD–#<ðªF’”Å0]MƒÁ¨ñøˆ&#4¡€€YÆY̯7=}³¿·ÛØ] Bb?™òzÄ@õp[}ô=SáË{5Ø ÷+‹•&õ<2»E½À(ÇÐþæßœ3¾MòþL ‡\Ö®Ô£©Íe¨*™‡4¡‚%6QÏ÷]嵐#<‹PC‚dÀáúìø\MÀ†¹xJ€wUÇml&VvV»#üšöõ­üWóÙÄŸ,‹ Ã.’®î#4]RÁ›WJ:îòô6%(ËÕÉbË "S3P›)Z=*楥Rm”›/7Á„¹«•ËnS‡tÞÞ»q×]©ŽtK•ÝÛ¤WvÛ¯-ÏM±ÊE-één®]ÍfXÐë£ÕY­£#ÜËTÔ¬îÜÚé·M¶ºm²–6ºÅ6îÛ´•%o.é7Njs3¥¹"šÛ æ®²çl¬k9ZºîµÕ¥j—m¼¢Ö—ø»oX¡UÜÐ#<XJz ð#4˜¦°R*fàÒ¸Dõ‡aèˆX‡”#)äpwÂñ`Äj‚ ”#<'k,ˆŠ.ÿµÅQÉϯ5ÇöÎتƒ×ÐYC¾ŠAb¯ßL{¼!=A ió«eŽ(|:š`ÉVÁ>ÎñI÷PAùýÜôwͤ:û¸Šî$LÌV£33Møëëš«êZ”D¨Á"‰’–°]ÿiOº „6|~}–Ê€°‘B $bĉàz£}#<Y­A–¾þ]£+›]¹Õñjï;—9K6‹2åoK^[ÒoÕíäÚi ÓBˆgÄ!»’ÉV€=?µ¶¾ù|[nmÒžëo¸bA;¥%$͈¨Šû6¶úV®ôPƒH(ÌŒ Tq#IãÐÆ!F¼ß­zcV·]­ÛfÏÁ·ª}K¦®­–Û³"E¶€AÆ‚’á¨é?£jà&!J|R²¯-ÔPc?ߧâ×­"#)´¢›˜Jı…Ûî?›ù&£×X¥Û"‡g@Ø™ÅÞˆíWìVØÙU5lÙ¡”ªÈ©#)@ ê®yý¤&碠æ#¬!+ )FH¨²AûªêóL‚(#~kü¯ÑößS\Ñté©-37øš1H±ˆ¬B88î8"á#<€TZ¶f—¦wQÇ]HíCÓ$(3–€;.P©\Ï@™~'ˆ\jíU*«hk,l÷p¾€ëz=7;á—ýS×ò_ÄÈÁ8© Sȵ^¯Ïü·èNX@@еn6‹£dÉžE¬¢¤iU€‘»Mu ÈfÊ+åw')#4Ý‘E‰IHÜ“ƒq8zqÑÉ®º¢Y¡7üõg$üôÌ*é…»ú„=$¡‚vÔríV°"_F‡ö8ÈþL+°pØ ™ ‘,aY$¬QŒN< EnìnÕ_nêŠõvc®I[G×Qœ²ÈH‰ím.!)B"B06ë#)lO[¨Tnµ© Q@1¤*Â3ŒªÝ²ƒyŽ…U…¶ò-o*ç?‘ªóÆÜ©-ËB!$-’Xɳ.#)´~Çg¤’Ú ‡©rq+U|\Iíö¼PîÎÊ«0þ`Òh(#<b@&©(2jp¦ ‰é˜Y%zþ!Z5‚ÃF£}uAk#))Ê )…X39«LÜÁb lAi@BDVADn Ôh  †f€‡ðþ`Ö°&ç–°Û@‡”#)ÖRnݸ5… hÍ3&Ë€ô‡æMÝ#<ˆ Ã†R {ÊÇv§ _„´£vàÀŸ¦ ¡X¦#)Å4„5¡H$b%ä ËÜ®ï=ËU0ùƒ¬Sùa›#<‹î° ð$c l œ•:öÁ¢¢²¨#)€@‰‰Pª’*œB+¯ö‹Vû*ˆÒ¿TišW_-;b¥D[0lA~#4¬N’î¶$b+CÍB‰s¨xOTMÁ’y“£‚$‘O‘OøŠ"mxó€}?zoXLѵÕ#4YJÄÅ´BX;¥—Œýðãa¿²¾æ¸]?ñŸ»¹¬nŸ~f/çà–CCš´H75Ì–ÔÉ®û®íõ9¥Ø=×{»P˜à¡ $jmWGQ—+mÙ!+¤ª±‹ý¹8fÙµ°"­ ì¼¼£’5,¼8#4T*Ë«m°¯v—(µO¸Ús^o#<Yç‚=ašHÁížD<ú7†Ã3¸<È9`TõæÙhÝ)PC^Ê?<£¨ˆ#4ÁI´å‚›d4P$TS5 Ezèi€ »=’Ò…ˆ…Á#<Š‡@J@ó(|ŃÜ`Ø›ÜÀsŸµ_ì0‚tazònéßõÛ÷ÎX›7÷!þŽä}¯Ý¿n¿ÙÒ’hÁ¤6#<™¡'Ååcg®9(‹¨>`§ê„  p þ Ï«³½Gàþzx·e˜õônœ®â?¤iDÆÛ×Ò¼~íÆ·¸K.BmA¹æ¨¡5!!9B4Ùûêôaƒ€O¦+Ž—WØ}¸4™¶'Þ’ÏèPé-¼Ø”j_5;DLÓDñ#)N ±8G-V$Ç ö&qŽ9qÜбws]<#»2„­éƒÃ|LX­Î.òÍ5>«.OC2Í\$DtÐÍÄ#<0#)×AQ®”64VË|²Sóvˆ÷§zˆo,åÞqü\\‹««ÑUT¿k…Œ#)Ø)p|èÍ \Y%Ž`aïhšÊ’B,ýn^^&®ô)×I“ÓÞ(÷+õ¼H‘÷Ýñçw™|÷šÁ‹øã‹N[†ŠªX>û-#<åû¸#…U7c™òw)ð[¡§g¥„JýOJÐ[×öGŠòdënR¶„%º-2=Rÿdп»Á¡1§.ŠX¦ŠSK¦•ß’ø®Àw§ȤrêѲ<`Æ6Ùèñ#)‘d #<å˜#30]Ì•¦!…(æˆi4b¦ƒ¹øÏNieÏÊç¶K½ß§têæøŠ2¶éx§kÏ<uo+„Ò[E(¹ˆ=g–¬éO4½âõ¤ó``òtç¤5ë<Ø wÙŠÇ©‚ÁŵÃŽ f¼»l"þ ¨[Ý×5ëÒÂœ hâžÕzâ8ïÞWržMü+0›¼¼¨Ç»ê5A5c6#<qØ#<4´»5‰t’{©#4îÚÞùMcATê'Ä ¿GÒ$Ù‹ÊCgš¹~²þ#<›Çú샬‘ÞÙ;3âüÂÕÁ×éÖˆÑÓpÓ#<À„C1¡Ð p‡à©a‹Hý+ €ë§%2 %A2íäxYì< [á„ëµ—±ï飾Rö“V„×ÑËô#<&T PnÏF‹sÈTüepŠè…Ë8<畸ú³pȾñ¯’v˜t¹ämLÛ#)ÃU%„ÚèFóÞihØĨ5ïÈj8‡È;gú÷fX’øÔ’Ð!3¹¡òn˜Iç;ªx+Qº%D¨Âx5$3q°èQ¿¯u716 ³¡1[ÎM—“¶H…Þa?ö(ùÝ,ŠcÅâðCušÕù ~Öoå[Œª©jUUf×omi3¤°gÜ„¤£l\0JÀY$ô²m²\ÕÎùwò/»Ü·¸Á/%q;š»{Z»ÄebÒ|«&ÝmÒ%™‡†¯¨n¿|áóÔ㉥®òÞ©ëÆQÿl^÷¦±—âX/~˜ná¯8mLÇ(èb*¶Ù¡AؾEó½wõ¼y3€Bhdž#<ÇßèGqg<†ÂgŠ°XÐ VŦ‚1|(%R&†1œÃ‡<ýs¶ÛÅT´ÏLU`¸@”áÌøK îŠQ ðÏsÆïN…‡b‘ð}Þ/;™Ä‰¨$ta+Ï^ÑÊpUBfMÌt¢Ø½¦1®#444‰Ï»F´Œúå¦îžVÍoHXËòz·Û,L(©$šqíߨðã…“¹lÁ!éö#4¯ÙÒvÚ£D˜]¶¬qÆDÂý=J/ö<4ïWó¡ÌÇ×û¾›0¬&ºžÙ@Cpo~@Ý€DŠHÈŠ%â## ƒßÞB!¥:aï‡ÈPþÁX8>Y‘$A$H‹(wn/WØ©ã¿–HöVÓñWœó¦#26ÐMýq ÙFö1ä½ã°ë£DãÆz|¬Ø1×õžôî—Ö@Wäj90¸[˜xÄÄ·nE¹‚(æzNÞ±på‡Z;ý»Î¨û—Cù›lëíà꽂Ó¹¹¾!À6Ý#)#½I_'ÎÀ SðÕµmó€h;4¶Ósšh‚"¼Šdœƒ”=àòU”;؈\NtHH¹áù«Ý¼…W\çL:qÕÈ(ã°â÷Qè²>ò*dhú¹ÑÊmfŠ>'o c‘0>] &8ÖR³œZ®§ÆycÞ(¦DaÄ (‰á#)¡¤`šƒË™ªÍMµi¯,‚üùq$p8…ƒƒô€çÖF­­w$ωÞ]‘ohÍ(4À°¬bqCåµ£ºœ-qú<þ­<þo¸·¨¡">ªltƉa;D€˜FÀ"CúÙl·{¥CÈŸo><¦ŒXÈB‘J©3„H‚–PÔ<‘wB›Ï}!®‰îõ0=ÄaÛýGC<NW› *š<û¯Û3?#43'1Aî§õIK–è¡ ÷8‘#<”Oû3lhj%à(ÅYýñ8eà’¬=ñnÜ.ðâ‡]âì­ó+f^È_d|K0ƒ1µý6·êŸ:Ó΋Qÿ¢ÚuV²½‹@¨%sKõt¼uÈMßcëÁÁ1Åçc&ÔU^͆(<âÞCž4±’Neˆ“V™s† ×*ÓR94P÷Ÿ+ÿW׶¶¡ éòÓXŽûãVÝv¼66 Ú£ßüyy_Ýèaz*‰Œ‹ùb§(ò.P RøeÖ%1Ütê7•ÎåÆš˜:p@/¿Nã‡#)îuÖuE5Y›Ÿ š`'Õ¸¹|{Ív­7šÿ#<  $ž­æ|lƧ”nŠG Ÿ†BÍÓ²µ§{º‹º6êš”ûºÕü›úð<yCi HaÌ'kQFIdM lé˜|2¤ØÙüx®±Ò¸Ãµ8tÕ—æñ«b>´²a•+Ðîðr ßb,–#<jzs®UÝ?Á®µF5Š€‚ÅXB *$RAB3·vË ¹åZ’HÀ0܇5ÉbãÛ•ÌþBîºJ)¿sS9lTX¨Úoíÿ/ñÞøoôxöB£2LÑ’"¦ÒÅÒÐLB£BbcFQˆ)²2lM#Ad©²l@’‘$èºôž“Ëy–ÐDÇyò²ëŒ±T>"nÉòC^ú8ütÏ#X0pœfØC‰Ç.:3SaÂ!Lj#8XÖ.oò#`!` {:Å•þR¬ñúOÜg“ó!@†4*TÚ|—^Y¶cœL T¶¤R5*³‡l½©[Ïœäþov^ú=Âi”N·´˜‚hbcm¼3‰˜Þ— óÃ`-®šMUNB7TRzk6ž^÷\qKPÍÒí®«X,Lý Èìoá²[hM[¥Ù<ó¸~Ãò÷ôj‡*”G#Kó¯Ì1„û\kF\àÍ”#QÎÖ<j’Aÿy2eÿwŸôkÆŽ÷‡Ì…AÄS–#3ùQ@Ñk-‚©Œ”ª¶Ónsnuo+å›Óao†Z5k¯ž™7¤có*M;¸Ð¤ä#+š«µÝ77T—UÒ;í{½HÆI#<Q£6‚3|Øc7˜P`µL)†ƒ@2 9 ) ”1€§“D@†b†&"øTmÑ#Œ(±õJÔK}*ñIIHFÓ6ón\º\³ºÜ¾–ªªpåúÞÄù1y5•iˆã>O#)SDHE²l*GSCDÈŠjÁ§ºš.×"~mDÔÕeÄÑs4Ô°‹þNWðö8YâUl3I±XM`t™˜Ä« AŠ(ÒšL3Ày1ùD¯è8Îq#4£¨R¸©£4¬*ÕÅQH•‡hf(<‰V€ökfÍÎ#Q‰)"E„I6 ÆF°A4ÛÀ¦¤#4.½ÎæDˆyw¶ÝËH‰ÐU‚õmãDH#<S×aƒ`Áf‰&j•„Œ>E¦0\ºŠnŠ‚cfÓŒ‚qÉB1–[jiµ4âi˜ÓÄ@Y­lc`Û†‰hV¨Èà²2¡±aØ:UªŠÄܪ´:R\£³Û+­â„Í°Š¸´&®¶ç·vã»[z¼îyw½$õ#<ã"e(l¨Þa)FG pn(”5Á¥êË4#4ìéåŽnãÀoÇnà»**æê®Iv´ ‘3ˆiR #<žžó¶²¼‹k7¨þLc#ÌÓ·˜_Trô#4±ŒÑ»ËƒrWGFè'bÌ¥+H­;#UˆË HVI†DJL£‹Œn—‚¾°ÃÓÒÂH-÷>²Ôšð)¶ZÞWlÄÅI°• Æ 81[|¡k€ìD#4ê¦mÒÌY’#mU©çpfØàB ѱàŒfXVƒÄ™¨J#4ÁŽB•|G1- `žiÂ>²#4*ñ4§š¬Žá*\`V!¸§daYFùNcZN0£C"N‹p."€È®Ë2¡‚h2Vj’RdgH™ÖŒ¡¤tRP M>ˆž«&ºjb%¤›FRÆS+j´¤,bÁi½QœU RS,JW:Ô&c&‘C#4ÈÃLæä «0)…$c ]B0ŒMX«A r ƒF4opAãEQQϾiè8-Š¾Pƒ6ðÁ6ù“Nâf©„[1#<X-Š˜A€c±Ë7ŠQ¶ôÅÔ>ÆŒçQ&ÆO‹ÑÓQ7¸Z¢é8ÃóhÍÅŒÓ÷Ó_ZÍš5€Âµ†rb&™¢‘AÕǹ”VG§•¸5ºC£d|Ië#[hZ¼%|Ïc#4¬åF&ÉwÈEöºÓ{‰a R‡Á:¦$ gÏÖŠ •€™B#4¥TEXëR GÕ:×\›%µEù÷HÎƵɬbˆŠÙlBá2"iŽ?F¡8±yºxÜF#4UD@!£DLìRš" ‘¤´ Q%Ì:êUeêûjöÞkd¢2k13X(#EIƒBFÄÅŒ®YhŠH†(. MjP®¤¥Okƒó×éýK¸ÜÑ®ó¹Ãwuá?Ï÷zéQ“k¾ut·ù½q>\Š¨5HÈ-njNðD@‚À#)¡jlÚfË&M[ð¹k½Îñ.¢€ö¡J~´?ӊ𲃠§Z:¡SÊ~ôaÖ‰QE¨” ¥¿(”€as»Qýaoºô•}øP8GHOEYÇô­VÆI'"{ÕÈ#)ˆE„HÄ`$‚¥€hDK–×윤aô¦¡LÃÙ_#)a4'7]>ë8ÀÃJ‚Â@—"´Œ‹íøëÆÞcrÿ'”5g£v´™#4¦†UWèÅ>IÂÙbƒƒ¯OóÖùó¸årî툋k««”’Ò!†Ù$#< 3£CEPl2˜Ø<„­ƒ7 Äq"‹––Sy$‡íU²äØ575Ýš9`¦4rît#ˆ8K¶Œ4.|ßïû–ÂÞÃá°“6Pcìï%p—9Ëí³]y[Ã[îÌ"ΈÜ0|^{©šnNcËŒ›<ÍX'¼5#)ÕÕpnáÁµ’÷©™á|ãÀYómÈàûµxf•éDJØ>•W×û` ¬BÁÒrŸŽ¼L•Ã. ÁC#4QBC ¸¹’V[Îàp“ÑÛù:®›ÜÞø6ŒúŠÉÐð‹ 1¬‰A.Õø~qO‘íSéÕû{#4Oç>£ê,ÇêÉM½¸<Cô $‰GÏzwçâ#<9Ñü'0õ»X xê’O6¿'d¯|NZ‹#@¡¾ßœ#4òbvš)Õ^Îþ¿?;¹'ç½HXµúv]Þ¥ügÏÖ~CM";neÍ\:I¢‚T ¡[þÄìS,Úš'„…Á]˜S pª&øimHª°4°—"a=s©Õ,I§iKÎ*¬mkØV;,o(@ü~¢Àöá*2¶’1Bä´X< °âX2h¸Ò=Ä+PÂLR‹d ùZ¿©jù¯»y~Ž.âaÌ‹\þ¶U¤Eõ¹ØÞœgCFÿ<Xà³)#4\ÕR@—mE„ýmg‘¾#4Α9`8™¹±jIÁ#)¤„9 ª¯õPÀÇda¨‘„Z l‘m6©(¢×œ­’Ö-æ*݉ncZ¹¬U£G+šÝ6¶é«!¬TZémr²k›šjåNï¦ähÚ“Zþ¾â5 L$j¡!ýKpÃD8‡swIDµƒD¾ºEÊ+ a„GaF%g¿=øbÌ„sf#"—c¶‡>lc9ú‡ÀÐ×˜:‘q0¤qA"¦#)Wë #)d Ü(GÒühÁ’EÈ€T¡’˜†Íû¸U{¸Ä?o‹¥'yUO®Í=#<ŸW/Ä Q519šY}ÓöÒ6}Ò’$ÒsAw´4É.ë#cãë—ˆÑÒM D¤¡$n" `Ê*¨ïá_:E-.©#‰¢‚`Á$T!0RKm-vª&«t•¥´"tÖ­äK·F#)”˜‰v÷G‘Q#4Óf¶Û~”mif£hÒk$j@@!D#ùÍþ'ÌBá©Pt#ÿZ‚z”OÂ!ÕÒƒªÜÊ0µC C@¾ñ¼‰B?D"ãï¬-õ”>>‚â#4Òg#)@Ü™ã“öÉ·ù­ŒÍ¤Ô)°–Ƶ¯Ýü#4ÉX†â28 ÔubWÒ¤ª£× dæPž¶”ª,‚²#<gŸ2 B‡óITÊm$„³CŒZY ËF“bÍ#<Ó4Z”¤É)&Ñ¡Mµ¢Ú‹kÚ­M¥i•J4jeMµ‹ci6£S[ßÓ|þ_:AÈ9%a™H‚4›rcAÊäHsŒ0¸ m6Ûò¬¤##q†0ƒ/âñѽ1Dî ÒTëFØ(„‚` 4@Q³ c,d&Tݯ,Y¥É.×Ûç¹ño2_;ÅMŒ$ƃ F l A°)ˆQ@Ä#DE¡•JA*¨B"Zí0_PTj‰ÕýØ`„&}•!jZD¡MÓSwtîÌ\;Sü…v›ç[«çvk^jº•RÛ4–Þ›éæÔÞ‡hn* Kˆ»”e„dùYûÇ?EaéOÇÆ_ëõçÙZ\òÑE*ŠW°¤‚¤ƒCØ·êØÙÜÐ÷5OСäØf(ã¶ñÔãgÒÒ?;C@K.0ˆË%œÈ whÄPÅŽ•`‚0‚ó×X… CTàýÞ~5—u˜ö4“HÆÁŒÙ¢içXB¤ê+ÌtQ@=?Á ³í‡d=P=#)”w­¨(Õ0 cU½ë½×nèí-ë.Ê›Jf@ôõÃám\Ôm¬Z-ÖѶ×-lº#<·Guu/•»`lA¿'±Ð1ƒœü!þI"¶/Ouê£Ò[¨†R°¤¯£ú´Ú²b ³†uÅ#³”úË „¶BFõ¨û"ÛôÍi§NÍ›¨5µ·¶} û¥×u å h9Ì âÈ¥/¢Áß5š5‘Ä"ºi*lý¼êÚ]¥u}ó%¬öxM³òABß~·;ÚÛÈZŸ<í±Ë%ž*£Ì&ëôê×ÜׯB~:ñëo¹¬ôBßD¬¿Zš×)(IÙ´/ÒÇ߇ã-‰ãÛ'¦qaÚøâ }¦anTïŒÂûÃyøD $ñ— lAÂ$òDPê^:©Päùõž–ågÕÚ‘;ws;Žú åwåõ‡ðÝêÙírñPâ@:¯|t‹ÊkF@ Ñ·¦¼,fµ†™¶ÛÚ:ÉV`Š¥›1/ˆ] uƶݙš N˹ß4g«ãYõÕ[ã¹…Ò ÐæcÉ?\ŒÏÝç üœÜà–P4’8ŽŽÚDñ§°@m³óû½Çm‹ƒÂ©³J·r½3ÎŽ’Í"®”¬"x¦± 6UíJaA;Jƒjž¯é#4k–ËP„˜ðò©ö\]tÆóÎÁ,ƒ-ç–áZ#)õC;uô÷4ÒÃÒ´öUØ2NC‰××½-“k©IB–T$‡ºè¤¦C˜†© ²”ȦäÉ9@J%ÅÝ?§ûqë'!Hå7ׇ%u<µÂg§Ÿ~æ×uî¹—P-«RÔ|"¹Yþ]ƒœ-öo¾N %UÝ5`B›yÛ¢ƒj¼X°t=7ˆJ›’]J DãVk ËJA†1qš^8È 5Ý*P…l«×®Á¦)ihJz% ´êV—:d9…&K\ß„ Sf¤~u™=ª3K6S<¹ÐEç#<C”Z/ŒɨIàÐb3°–ü0¬.œ¬Á”‘Eh|¦t£"P ™jš¹ã$¶:‹Uµœ5&56±Á¼©he$³²ù÷Û<+;Ë›kiÈMÅsR¸¿¦ûCÀm‹f±“ 虘·â®8kÃÓ[é‘þ»ˆž tý™ty[æâ6¥´®qj#Y«EU*¨M9Yžgrª³o:,æ¼7ÂŒ#4вc: | ;ˆ/Â96 jb_Ceƒ±¶æô6lÊù-:™œ3Ç­Ó<i6!ƒZ9›•–bìRѤnÞ97Í`ggý\z׆—69…{2ìKí…¯ ìZÆÖÐTÛiã‡3Îÿ#4¨õÛ yã¯yNQO$“ÑàÌÓGÍq§Ä½%{f¼q‹‘%Ö<`hP²ÖÖ“œQ{îŠÖ3ä-²;ŽŠÌçseSÍ{e9jÜÅj»ÅhÄP¤ô^Ä"n¨í±Ú2˜Ú–ÒÓ—ý9¥éˆñëßÇzèûúŠ°YvæñpbÌs%.ýœÙßjË´.šü#")cÊfGwû?xñïé‹Z¾ªEãJ¢g ûטÝÐÄÀ¼ n³àßãxÉàŸKÂ81Ål«¦uÚŽœ©Û“5Ô½|pÍI5=ù‹“ÖtòÉ#{5é1ê»âëÉñ¶tܧY\óÑèäÅ&;·0¸ð1²!ù;\ÂÏrï·úA¡“n»¦“/v]šòô¿b•#4«¢Ñ‚–A#<˜ƒBÐ~§4rhj×z¤f'–@?8 ÌPA"/ƒ5ç9»”r»B ŠXŠ6Æ¥-©¬JF]5&#z^.W™Ž™å‡½`6Θ,hŒ/YÈÚ–hL¸1G}ØëKèMzw—$Ò&½Þf|öªaáÃC˜ÒŶvæ±~ûßBs¤ 3%·R\ »Õ2OŽ›í±¬æã뢣p|ç{Ë)˜'Ç“ösÖ/Y˸¤@^§ !Ðã(°Å!¡cÏv)´#40ÛkwFã1œ¹¡¼XðÇr Œy`“(Êð^XF\ñÄš*†„“bäjЃíÚ:(à74¡V…„uÃ^8# Rg„Òp3â†)‡q¼.Êl¾ã´#<Hmm®ÚÌ…^fŽ¸€vCƒã³WÛשA° [Ã#4dBšƒ†î[¨Û+•k•zØ}mäëÇrQ”“Å<Irò`ÜrFbHu ;q`5?[<ûq{ÃÞyiO®b-ùa@ÝU#<3»’[¢Ôn`™¯)®C3´ ©<u%‚Í*%0cq²#)B#)),"¦°½6›áÏZ®A~{k³vxã·;Y¼È!iféåaZ6#žó¼%J+¦i~³E33yâ‰q×vÊ) a²yLp5kÃ~­N6Q…fçtæX*½©…&[q2lòžeywÑGïÈrw«¸xëR’q@yK#)·mShÔoî&í· ¬!D’Tݹìá&Q¤`ºD¬»O|[_ ààŒA0pŠŸS ¦J…¥ÅΉMÆbzwR@Ò$AE9,8:À.¶·Ñ)}®PI‘¼¢¦—ku°W 5Íy 1[mů0zмmr…‡í*(ESiDUMM9qφn']¦@Í 1YÙl\oŠjš‡ƒi¸Úì›3œ”\å½dÖøâ·¡ž ·—dç>Ãe5#ËÇ@†$vn«K'•Í5qÓé†Ë¹&éœCR¥/L#4Ñ1-†"x,Í?‰Ã‘#<,ÇÌÎ:ã}/§…6#£àvئ«3ÅZ}¡tËh¦ïxwu®¤xt¸Ãìa?‡.pn1YVAHÇ_JR›+ç|zã¿ðüI“c©Ió¿ŠÉŸ‚7ZUu¨’ds|>òí‹sN€ƒŽ+b¬FB•}]°E=]”¹Ó-5­ëAwëˆUÙ¹gi;zu=38å#4–+ž6§ßÀžiòçjzñpîššËÁ×0vÂht.뮲Þô¦§ÜÛDà]zÍÓSqæsåáƒÃ¡Ñs×SVqٗٻ0›¿¹ç/¿ƒ‰~™¨Ú¦ôw6æ¼¥Òâ ²Å5é];Ï™~/¨4wBߢު|œ4AÒâñ’MvãÝƆ8pÈ|Ñá©_8Ùóó_–éîçßäoø}¯"ð˜yù?XÂUYî2%Mš>_YëÉ¡‘)¨!tÒÕ€i‰(~ƒ:Òt ñ‚¨†–òÌY°½Säy~|ú†ÃÂÌ,:¸ÊÊ°6wÖU 1mrD@f¡¡5…ƒ«`^†å™#4î›ÒŒ·§˜“xt4|äý‡|#)Éx×€ºOaãgJ»À[ Wv-Œ3–7Õ¸Õ²üñ-™! U‡ž‡ hGŽaB«ÜC=åË#)Â\³båÏYwqXÌ7·{&› åsV¨Š,rblm¸:ôç$Ç”,,¸Œg ¢=hªz {»–ʤj®¡PÔÛŠBÚvæ5…änT÷Z­%9èù÷Ž qCLAR6Ÿ‘…¢ ÁfrœQ ¯X)â#1dœª=‘‚+¡ºI¥ÞaÀé€Î‹­NA#)`Ä-s#)ose  =ßA~Å]±á¨ÔïÌ0Ù†n‚·5žtò¿‰N¾ÀCw¢³0ñdRâŸÜD ð±aßÉÛdØÌyååÀK{ ~Ç ¬ ŽDêÁžOÁx#4Þ#4þÑ ÑÅmýéRª¡§ÓùN¼$Ñ "^Ðïö±{ `qC.“ÐmNÛUªI*këßÓ™Xš¹ô«èZ<àrãéZsTóHt§sípqÚ|ŠÚ£²*“J„lÒÁ±‘ J1~3uÞkuƒ´©ìÉ¡é·tœC¨rH&5¡Æ©²Ø”CÐbj„$"Œ"›CISçW{|»ç[›|ksFÑ©)6‹skÍé;®I“iš¢óuóÊSv%Jˆ\¼@@#<"ÒaKAEFñRDV ‹QG‚²,PíL±µ‘@éX,m5€Ôà° ªQFB¥´ÍìÌ<Îg¬æþóš~wnøa#4RòL#)nmÁ¿¯&0-—m%ðN²AQŒé6_Ë8·Åo+yW#’Æ´Úµ¯à¶‹U[W6µª¯—Šm›ÖŸÔÜk?µ#4&©coƒñàÌϺ怜Ë-N‚•_ßëcÆÐÒÅ#<ÝD2†º}p£Ä ÓmÛ\ã Ü#4hÆèÍ?Ö:›1h‘]…2¸™øÍO‹7“lTë[q†¬Ä3Ë+«†#<RñÞ×ÐK£!#<HR UÐe!Ã)TE|¸ÓâopÇ­jê›Æ³‘, ,ÊŤÍÐÔq͉¨Àm6–5$RÅ.ê+M1¢@%rÑQîJ ¬ä2mÇvî™Hì’B¬—Û-÷p´Úa#<Ͷ‹Í²Ù0¨)d¹ 2m špb#<eŽñ. S5—¬2&àiØÆÝ{S¢#<C¦3ŒÍ…Kܺ‰ÍMD‰“ÜU­=6µ­S Ü!µ—IÚÃZº¼âÙη–Vñ“m¤†6X*#<;FÍÈ‹¦”SJ†íM§£4FL16Ô³‚m°É3©kôßÔëœÛµ¦²œ§x•¹ Ž6 cÝZ‰Q»%q1£ìx«¬bÕÈ kU¡¢–ÖÚdP‰`M,Å7—2¾ªÇ&GJÖfõ±ã„Ë©DÆ ê›°Ý!¶:Îø­siH“°N°Ó†Ž;j8õÝñÃ#)Ê’j”Ž6Q½Œ)28 F0›áÆ"ŒœA•'#4[ªˆkHÐÓ9:C%i¹Œ!Ѹe)¥#<°È4n#<hpÒM615Z6óqmäÉc®»zça›[ ¹tøÐC :Ô6EJŠ×gƒCW Wki5a”j¬æ˶›¹Œc1 ìs§?Øyo·n!1Üa:åñÕaB#i¢:š§ù2²Ä7$~’^U*|ÜtëS#<˜¬Ócgi‡LÇvO#7Ûö}sj´2I Èø³qÉ æ´Ü…¦Ô°Œ!‚ÇZ1šKZ&Æc\˜O&g6Õ3s0táÃPQ(XS ÚâZ¥EÍ0 Ùé¶Ûu¤AÌ"Q[Yý¹J<AZŠó aÃ*Qðˆb,`cJ½È©„QÖÝtÁE×xiõ`n À|’WŽZhxÒÉÍDáé·‹ fR˜q ²@i­X3;¦Þ˜2âO³°!ƒ,„c±¦xw }Éf·ºjÔàÜ#42j“#)Š…™83V#<æebkwZ!Û ÃÓ[±vJ†#kS(i¥§ŒÄ˜ÈH7γÖYÀ^D(@`“dYr •&º½Í²Ù·ó`yÏPiï>Iƒðj¥20DˆÁJ;K.ê &ÚñÉžŸ^¿é¯uJˆ1Ë&éûí_Ûe&þ7‘Úë:ÄJFU#á&ßÁ(æ‡kÔ#)óì}wLH~f44[’™ßýTwlfƒ`Û ·7ž£`oËóWòþÛ±Gó¶çç¯ÂäýïúîooTÝQ­Mí³“”“gC©1í»$žã´+Ü)×>@*§ÙÜVêž’@ÙY^– H%Èéëéú¼à驇 5ù¸'šÆKdáÝßåU‡ÉÝ‚‘‰!RðÉ·Cl«—Ø. àöz_ð•Ç«CUDTÍË-¡Åå0ϯy½Ï]¸Þ1´5 Ç$Œi·9;·fÍÝϯƾ·îdXî_¢ºmxñcè”z@âv„pi8›mFrFö”Tª3HRCíº§ØtðëÄÃ*:‡¦ÙåK! Ô{B.O#)T0Š/$j:òÕͽ•yY+®ºÝ*ö©”Ûo6ɪ.ºêåb©Ýª‚Ò@〘´!*@€‘#<ø‹«#):%#4dž"<LÒ‘ýúçw][yø‰4F5RM°¤¶¶Ò’1%D¨šJ©–ØMcl›ÛX¤¢¨’²FÍ•Å#-#<Y²M)6JQL¢*#CZ#<‰LKFh¥4RšE*”a¬“#4°ýk˜DÒFATª(’DDþC‡Qç+Í0A˜ÑÐÚ†"û"qíç;YËiø¦Ñ}šîf·p ó8º?(q$„ƒ7÷ˆp ¼¯_}´Ó Äåõä¨>÷–jGÙŠ€*µ >ƒõª8e’à«,Fœ`ÃÒÍfÛò#)ä9F×µ› ´Ò´â@-Bj{(D™®'ש÷Þ›nÛý©wäwŠ $ °U„ÓfÕð1#4æÒ»N­ŠàæÜ‚#ŒaK1<üeä’Bèœ-n£ÓèòÈ}Úº‡kÏÔÄ1TÊ©J­©Ð¦¿Vl[zm“,dò+2ô1zºçk ’ÿpð„òd€#<&Ûb©DmIkEµk•—öw[•u#<MFŒm±¬g麮c+Óku©|ºµª¨O”,µddæd2CÕ†L°¤PaT¡’ÉgßíÐ[4±E”žL­*Ë®7õ™±´5±gÔÑXÌDy#<–Ô#4³‡òË£U¶4¡Ë‹§CªŠ82V”Ô¥VÆÊ<DP6`äƒ`ÆØÂØôÈVjl"…‚#) e’¤ hâ8ªai0çÉ–Å‹C4KÄ™ÙÊú÷k×;F2•Ñšú•êKÍË$Úë·j鎦“çËöàrÕ7ät\co(ûÁ§œEi¤¤Ek”AÈȬH ÙÔuQ¯2™dô<ô’¡ê—™D¨†Øú%â¡»T™m¢8D6^f$BƒæÊm~#4nÂ1+–£c}#ƒFuçDeá©ÁûfÄg›ªÞj’UʨØÓSyYRw|w©0ÈEÖïu‚#„Ob“bÀ ‡Q2JAB]U 4BêU&Bà&-@Ò†‚Ò˜¶ aP¶-¬<Æ>vII$‰Êà Š‡«÷I‚œÔ<Ó#))\2n‰i¢Â@#)¯Oke=âÃfø¤+#)û>ø»»ˆó”gò&–¤ó4LGÃO‚1L~ÜãòȦbîÒ3Ê kßÞ%™ÜMòà¶ì¤ôör~'0G`:¨‘ZªjÒåÂëF¨‘O¤’RÒ\2†yPBîDÛÚëïD÷¨‹¿v²¬šŒˆ¬] ßU¯FXýÚƒíb…4z]ã1Yøøyc*Š7ß­š™*¨©!žX…t:¥¨*o}Aa4ÇmE‚60H·# ¼N6Ú$OPØy|“÷ /]#4.§Øt[øèýWF‰ ¾†¬Únµ1Öíq±‘·7BhcÚdʆÁ¢‚‚š$„¦™(.ÅGÝ-Ëšé[ÑåiÈ1ÕhEõzO]ƒTM¦£*ꇉ»eß÷Kê}Y’‡hp±êN‰®k«¼ç¾ª$Š¬ ½XåÆŠ–àø ŒÛBH-)Q ©bùõ¹ðØOn®ù3>¯_ÒMË°#4‚š²ê9;½Jbb½¡(Æ“ÞuRÚnYG¼áÚøÌا¬ñ5憦âÅKK&^eAðkú°[#4"[@¨ï÷ü¼@“_ºibN¬yâ4p€Â- _»é1#4݃²©5™Ÿw?Êû4aìï2O¿îA„T ˆˆ‚¬L|ãнþQgy3/µûþ­ð`TCBÀó<®P˜.ëÙóüÜhk ÷çle$Øû3ƒñûãf¸îuÍ!UÓEˆÀ†H–cku`ÎOXÚv§F³Þ“¸Ö ‘Ô½7î`\Êx2rïÜ#)Alj)ÍØÀøÈoæq{Ó ¹˜•ü¼Z\úëu4î÷´L¾]EçŠ+ù¬*ôñ‡ƒc¯ÉÓ!`ÙÖ¥G9)Ûd³Áí#4½³Äˆ¾"Êòôü#<¨Ð‰JwcL]ÖÒ4¨¦g…§ÖuUÂÚÍ•UÚ„!l4ÁAh#)†È!HŠC]Æ÷¦¥¡#)‰M} lÚ+rKþ(QQhdI†ù)£Áw&³FúÀÔÌ`Æã €…2¡¦Ó<I‘E¤€i.âwp+X=0銰l|(C<ª/x…¡®6oœÐÒs¨NfÚæ fŒìíž7Ô€ô³r#4£ÂÆcdJq°@j2m%ßÓΞ‡uzÌ'|#=s±LaZÇR}IǧŒ0ŒðÄtÃäÊ ãeYŠÂH™ÓK Pfw€\ bA)‰š¡o±JÐÈE¡W”‰õÅàÚÙ6g#)®@݆1ÂFZyå{µ…Š˜D*‘Tòzaík÷Õ0ñ &6+ÑçLö×~N(]v©VÓ#4 Bls/, =C¨²™©˜W¡Æ÷3¶°Ò@Ž;S&õ{zÞÎÌﺆmÇL0Üc“µ úÕô™l)[ÅÁ„[!20ɨ ©Ô 3ÒEDÊä˜x]–vÅãðñÿ ^Ë{æ%µµLO\q2™œè9ÕÇFÙaÁ&\knÊLp²f¼û†ü’1Ë)¼@.Õ"T³[•”CX÷É<#4½ùp"± Ž Ñ#4Ê#<´††’õØk5bÂá„Ñ P1BV X°Ir ­#<ÕJ8AÅÜd;ÝÑ”#<ô³2•±žû¬3RÈcÊF][¦ ‹W$««š;b"“ ñ2”¢Èvu*9i#¾µQÀšG¡RÆFûf#œêÙSRÛ´XãE©Ãð"œuìô°eA«Ã-s»å‡™D‰Œ ã Úm”m2€t 5’æÈ8-§2a;HøP°Rô‰-§I†Ò¥†VÎèz¤´·¹Ó±QšLÄš“#)A»m¶ÒÁL\•#Ð×:y1–”–_;´²}NQ [`ØÉ/¨ÈœÒ䘆n#4±‡#4å´†„Òžøq Èö«¦Ä6È&ôð-Ûm¬†!›áC³±vy¨óŠl‰‘dÙ.žõeÖψ[m$KYq(tS¼¬I.›ûµ$[!uvÇyˆÎÏ pBsö«ç´H©`‡ç‡­Î14ª#wŒˆyˆ†I-•â>£pz8°çÇWfÑÉdíp»–͈’_ ùdÒ-ŸlÉpte ûúQà°±zâZ…ªê¦P󉮽œ5Ûx'×Xªñs&âiÒtîJ¥kŽµ#<µ‘eP#(k—„+r¹Ä5­k¶øà×L5¹í7«H\•*8Ò8r"Ÿá—¥ä[ò-láy’³LÛMFU­ÁE>ÀKKK#<KGEÝܸðŠëœ¢÷è"2pk“*Ð"ª}*eÇ ÄÂfÁÐ7§mÌtàÏÀaq³¦•..g|̶ÇXw0dÖ:¸mÍ“táòÁ@ᾄA¹%qÄHFÌü¥Ü"8×ÚWn•4i²£¾ËXÂÚ#4ŠpI±H&Y@£8ÈÓM ©#)Óilb,(B£#)F£á­ó­Ñ2Ø8`ƒòbÄVÚ\ƒœI “—+ࣣYVHk¯¸lÄofMha’Œ/žøPç)Õc$:råî›väšsgr0šZÈç¬Ñ]lï(vi©¡µSlk—L“k‰m‚ÆøÓéC¤;ìŽSJ:R™7NIƒîëû>xMëY½ÐS7Q#4œ?–jÚOÏÏG=«}Õ‡ÔId0$ ÓbS?›žl¼:Ç’ÃÇ]<ˆ)±¨!ÜrŸùÀÇ¥¥Õ#ãqrÌȹêaŠoS³åÈF·$ÕMü†pòL'ìø}4´žžÁ“´xhÞL&GD$ÂKXrŠ„ÅÃM¶v+ÓJ.®jQë\¼…,PÐö.&ÇåƯϭѠl:j'F…ZE6ƒ}QcÈltÑ Š°K¾cÂ./“™#¦ðÓ4&Þ>9ÎT¢L†äÞEézMºÃg#4ðõ¿d:ÓõÂvhœ›ó͆™gæh˜8œ )2)åCB£Bh;xñ¾Ž=F¯gM 3Ð9i‰·dTÓCYC–ôÆòØÔ¾2™!ÃìùãQ¬ÜÌÕ„Rï;¦m¦8q¢â—&GND®ˆž^na¹€Œêš@„Ôí״㶯¨&Á´Ô¶¥‡O]nN2AtŸ¤Sw·{Û¶›òÔ,1ÄsÖPÍ °ZíÐ@±Z4½ªîüµÅ7ݳ½¾0o9WÖÂg•‹¢Q1ÕJVæãºät±±0l(]ñriî%É«³jhÄiƒS¸ÖšH©‰™ùÉZ|;`A¶¢…Hy!—«lEáA©f¤\Õ(c.â@dŸ>ZL_ 0àëÅ5£HÀœ!F¤01-·@C#<i¾„Öä Yˆâ\ÐnRØŠ4!F#V ‰3•P¢p0\`·LA¡ Ž!ŒC>ñ’¤Òp5PykaHŲ”R¢J²Ðj…jí#4 F#4VJ‡pÑ€Ž ̱U€LtÉXP¸âÛ35¯ªÓ6”Zí­Û‹ŒÊ5-áÖ/r#<iknˆgt’â#4`ŽQCÂ#4“ÃG`|ŒØ¸#hå&TîlÖ$M7²  0gA‰Pß%Xá jÂБ•rÛg&l9AÜ 0HÃØ#<š6v4#<0eÒ#¸CFD#+#)¡rMÚ‚ #4ÌkbȘ’MxÂœóÇoŽƒ`r8à#4x6¢éˆ;w‰”£tbDU l#åͪŒ#<°ˆ[bf4ˆ¨k¤m¶6“µi‰T0§è ƒ»åç{ üG>Ç#<ˆ¨2Š*"( kÕ«úøm$8IÏMñðò“Ô7 Ç/w%Š ÂШšÕ! ¢È#)‚Z#) ̈vþ³Í®Ø("òäOò…‡Üþw6|àŠª‰õù¾²öíb#4ü_R˜k¤¶—âvÆúÈR¨äˆguû жi)e«dED2÷#4«¼ ÌøæfyÓE–›áÀÕ4˜ê<‚¦ŒÇú,#< ƒ`^:1έ5赬oKŽbMp†¹ç¼Œ› \¨C)ÄÓæ–kåÛÑUtËJ¹*©õZˆ¿ž ¶2Ræ1B½”ªÈ‹Ä›ªC‰4õÖ z¦J#<)g) ¾“pp§^üìc|ù÷³z;³ƒw)S÷UJ>•Q&vñ@Óq#HªÕO¾Ë$ÓÐgfJyU²ë"i“'ŽN÷l~ G5i£Ï§{èþW7=Àˆœã h©_¬. ²¥Å/³6‹tò7¸Ãnhûž÷=,ïf³–˜Ù"É…­Ž‹-άa¨A}jz«÷ÇÒ)ÛF;÷¿æ§>2îqäU³-m6DíÝÈ϶ì÷]#4Ù9Dæ|5bÌ"†þ­©µ}íø'vØ¿RöñÖRvuÁÈœaÛày(Óñ8Ófædk·W#)©ÚJуIFÓbBˆÁØã$Jðd™ ’S5ä´ï7&½»´f¤W{·Ïp•ñÓåÒ¹q&8å ÔrK0Ä$(,ƒM4¬´[Õ5ò{æë™%.ͶÝo­À#4¶¡ #)ï†}qgMÞ¨^•¥8Ö\†¸ï„òàifní5aÅÓ LŒ‘Ô«28Bˆ0dø”Ž#c|ôvvÎçÛßæ|a‡‰á«M¢…#ô@zAS’sœS¾d|ÆøQPª¦õ™c'Õµ¶ŠªŒ@ˆ4:HñÜÂÛ ‚­#4‰]’&ÉL+Ãè·ÎÔGRÍu‘3[8 A¸ =<Âk‘Ç @ˆ;˜Ž´x2Ó½íŽÒ)ác·ã#4ù{’d#=›c9ùÅn®ïº¥Ç·Ò“Ñ™ÑÒEý™7ùŽÆ§Àk6M#<ë#<@Ôá6¸˜KvXaTN+ÎMN8ÐÕB04ïîÂÀ<8мRÐ;ùu#NO#4"¦‚3OLì™o±5?TÌ›uŒ›¹a¸x…ÇÍ#)ñB60›h@EAÙe|-ël룭E5ua«Åw¤„ ¹à.Hl9LÜ_¶󖨲å¤Af1ùøù1†Ð^HÆ?Žß_•›ÍÅb©¬DõÐR?3YÇ éש¥3¡FM.á¡À“ÔâÊDìBQ#_½Æµý‹mÕL¶‰(Ö6Ͷ´¦2mdÖ™_Ç5’Úú)#HB™ 4 Æ9D¹´¨©{U¨²"Ù ­Ò#ƒDÍUꈎåžÉŽ˜ÑBÒ¤,8ÿ›#)#)=Š‚fköi\½#4âXwmƒ™Å„R ‘@X‰!ª÷ñðãñ|‚Ÿbš”-Jﻆ@¨X0dTdL>0ìòZîW§}%;ñQ„kÈú…¶°ÐÁH¡å#4¡­CÖ;×qÖ•ÀêůuBN•s \à 815 p‰5 ~‰ 4a£eŤÑÂ)#4…j\¤j´D#)Q‘²¸› áQ"d²:šbt-•cUe¦\¨A3Li‰„µ¹{v»Q¶{qØ­Rm±Ÿväv½Þ½6â€<Ž]oˆ(o*6À UÉUॅ 1,Àr3µ§Ü =ÁÔð=<)a 0«Þ^ßu3'# Ao !ÓÊŒ;QzËd’¿,R†*öfô@á–&îÀæöFÉÜ;]¤TnBX _2I%›uã¨tôÔ%ß2Á¢ÏŠ¢Íô0„àZYBeJÔ‚ „&QÄ×»´&#<#½¨¢#4þ"nïiá6X€.#)[ÝP4³&t—ðþïM«I,x{>´ál|tè7ÃbÀLtKž3³E]s¸(µ$”#<¡XÖÅF¢*Wˆ®ÛÑ#)±Ä¡ˆ€R@\ëc“€öòƶ¬¼s³‰ç)¶Läk×æðhô?‡Þ‡Õd$Qy¾BÃH,0Š"&’$˳9sÛØ)O_1ï=80$±„7È\ú9ú“ày…<ëÞóÝ,ß5Vßoçù|#45›mͤԔªYTɱ)C6µ±[0ÖB4l¬Æ«IW÷4[WÍtÆfÍ®þ Kóïã7Á6Ï2zc…Š®x™¯ËZ6ê!Àëé7yæ§$3@;ÎgÄj6ãÏ«1êv5€ç³ŽÜUd`A$AC>êsgÌŽ)õæ)o ú{6‰RN\­Cà‰Ú'¬÷ÝA0*§0#<Tv1~iÒHs>ì|zvLÓ;|=ˆû?Ù'{ÀTüjdpËÁFKÔÔ´)¾c_gŠõÔä5„‹Dí:)éq„$láÓ¹0ïÂÐ 57¥n(6äÈՒ˹,: 6Pjàe(liÚtá‰g| 6ÆÈÅÕX ˜s$š¹Œ¦s«yàó³—.ä“vN#)˜t´áI¡%-[â¬qnD`Ô5åb,ËÈKVZŒ±+…í+mäÆÛS«'àmC‚À€ ³,̘(˜h…`°‡43Й&ˆ@Ám‰à~ŽßDTÞw d`"ú‚î¦L•Ø0¢í(Åå(Ö‘¿Ö­h`}/F´·XaÉÂ_ÀX þdÎ…a.ªnÊBÍÀ³#)¸äEÈuöú¥,–{Šæ³»š0¥W¾ÝÿEÁGõ1"(Æ$`„&Z‰ì !áÈÅ:ƒ¸“§N˜á‰Ñ‰‚exQ mCÈ"HÀ‚À‡PÄQÈŠ‰µßwKÃÈÔŸ9H#)W›‘¿¼ä%áhÔ#<Àî&Fª'‰ FqY¢™D¯W·ÌûÝ D¥1Èe è:¦$·¿c’•íÒ6/,ÖlŒÕPª½I!’0€d;Òïêçò“Ÿs¦¬ØÑ»lvs#ÎLÈkÑqË  ýçàRo÷íúx{vÕÃbËËw§éÉ¥æ'¼xs·ußØw˜ª‘œüû´¸Z§1˜#)Ž@0#<m%›5ï?¾†ºŒr^ï0\é÷’- ƒ€Ò§Ê#6Ñk#)#4C²(«½G¥6²Â#)ÄëÛ¢À,ld $Hd£3=×Q&ÄhÒnuÝrF&E‘¦~n½„Vò×7ù2ç½{o|î-§ )(ÚRÊ0ÒF(‰znÚ,èÙ6- ZM¤C&\Ý$–l­™­®–™óƒB¨ö­"šÄ‚!„¡°¶ÍOißá( ¢«âWÃÙcÄÀ½™êòJÈóüïœ#<z#<ºƒŸHÈH0‚ˆ2)  È‚#4Á{Í£ãÖ6¯»õ‰SbB£R‘±´i#<Y’©·óv¿‡ó"£~Wæ¿»#4"(щ±°ªfØÚ#< ê€|}<1˜1C`Æ'Ѐ #)Z"iöBR×#4ÓÖ$LŒ@‹È†*y˜-Å2‚<*HÅÄœ`•}Ò¾i»®lƛȭ•±E±³#Æ[M(¶h1>7ž[i6Ô‹5¥-[öþhKÀB.ñ#<øàã~Š©¬ÿñôÑ×ÔŸì.éÜI<@<‚ ¦Á£½}'Ÿé®¦VEMW!aWÙ®’b|á X7‡b!á"F=¡«#)Þ×™\…ó˜õæ%Eñ=H>|ÐÁ÷SUãoÏ{Ø,AR ¬"ð•ÚOãµ#<"( ó+(OD˜ò#4ì…Å„A€Ü?b!Ç¿‡gÏ!l–÷#4oZ¦°H£¬‰Š Ñ=ËP@P^h‰¥„E¶ ,Ý*@ˆ„#\ÝÕÖ’Ê-Lêl¬Zl‘´PdI š7Ï_”ßlßœ‚†°ŠЬÆ„Ê>4ˆ„ˆÃ…¢2ËÛ¡Vœýt!î•#<ñ×"hOåÃ^Ö?q†°Êôæ–ÜLf‡`= gvq-)AÏ/7©úvÏñî#¥ûXÑ3(”:­e­x¨áª-ò2@þ\ùÞôÝÖ^¼ccßIsb>²`Ëï#4„Àa¿™¿0})ýê Gœ‡“¤<ªF˜@’MRÍ4©!)lb¢ƒQT›T)h×ó¶ßÐÕæV©4ÍÄ©„Fxw=}¶vË4#4QdD‹#<Ã#< "]‹¹È‘+O¸0ra#/*)bbÂ)¢fkØQ YA(#4Ú#)yß õÇ-˳]èvÜu"‘þ#)BÆÚÕí´V£ÐV”5E„«6X¬mµM¬Øø’3†:zu7úÙݽ´ªË%$äkíÇó4Ÿb"€k54þÙYcDcxã1¸ªbÕX¦QÂ.<ðªÈ?((4¾¸‘ø›ÇõdU¤V”¶þ—‡šY»fX†ÒŽ #§DÆ&ynÖ6`äŠhªÿ¯da±¹Uùt¯/X¯u®›Î·à·Y¤ÔÑJ–ù›#<í§vmwvWšöõkªmKo\•öî³MÙ™U×7mE\íC-“,µzîÆšdÚîêîíl›*J™›Þî­í5î®Þë»UénJE„†" Û0(ØÕ’xEñ¾zâ*¥4¶MI•*ø­v·ºº÷»V¼Ö6ÊL̀ȂP° —ª¨4•M ‹#< ƈŠ=°ØðgdÀ2@£DhÜ@U¡FéÁ þ‘ÏŒ“PocpX²‡Ð¾d-‘QŠ ™’1À-±#)ÀD-Ãü{Úó7 pÜ#kö±mŒAV/†LÓmuÈâ@œáË}ZéÚìt!Ò=W3.F=ª»?¡Çù.#)wà\Ä~‹âTõüxÃëÌ=ž4.(XÙÆ0$6ÌÞóXn*SnêccѸÜK‰ ‘o/#íý…¦F$Ç©CUR§WÊtÎÒiMfši€Ê1…¾Ï#3\…; >Ý8‚H¢Ž+”R1‰x\ëÔ˜ãÂS£\È~Ì5‘EÖÀ’$fL°dZ©.† Fîé1=>[Ì]Yw~%‡øe$¿ÀŽ#)D¹_©Õô:}ÒV]×J“Ó[Ø7æš°íã¸6Ä#<ÑŽhÞDš}ÿ@q3ÚM^-òOs x …¡l…!’c2øU>3Ön£™qÝ8Ýd²Ù#<™û“؆.Ò†Ošò¯É_ª´’`oÒŠÃÆœƒÍÔÝ”‰ê€ýÇÑ(œY`j¦årÍ”­'S-‚­„&[Hp íª‹Ü4Lbu‰­3u0ˆ@ëÙ£0(~Äâ¶0íòíM{ :UŠ^´Ê“h«ËqJSJ1(@¢ë!Úºÿì7ø#4uÓü!V»Lhèä‘تÖ6Îpë —w8Úÿ£äædW”2¦KD”èj2äfæ@¡]ΉÊhË­þú¢\’äƒÙçrÈ£ºËdËU›xMv×e0ƒš¤hr~ô”ÄDVgº NÙìÛú%²AùºØXÄvŠ#4/íƒk¿¢ª¼€fÔDdZ=JÛ+xƒ€ÜUvò§ŽúТcpkcœÀÁ„åGÙ“¹)ŸžN‹¥Ç :D#DF»gŽáØ2ŽüDÇ"†zY”Ê-3\NbBC¡âÖ»ñ“pù™²À’˜ccñ4mÙÁm¡ø€…c ;Òq°HÈ÷‹ì#)Üt8/B ˆE0=°³‘Úy{s1 nr1ÊÖÜVÃÙ#4ˆÎÞ«~Nò߶ڰ ô´¤ûlí<b”¹¨‰>Ü „Ð?#4_Êx@ÄA6MÛðˆ,¤Aì#)kãÀ|Ïâ,ž™/ˆc*(÷žô|êýìÀÇáwÛ}ÀY‡]ùó4#) ˜¨M_®Ê­¢ÏòÿêÿçÿÇÿýº¿ðÿ»ÿ/þoæÿøøÿ§þm–]ÿóéÿÛÿ/ùÛþŸÇ»ô}^ߟù|ÿÿŸ†ŸçðGþŸúCÿß“ÿåãþòÿûÿïÿ?þßîÿŸý~.Sÿ_FËáÿútÿ×þÜþ#ÙÊßöÿ>ç}gÕ?@#êQû!öÏõÿ°üˆ˜ˆø„pLS"®Ãû¤U!Ä*Ò%GTÀñ¬Â%jĸŸæi Oõ¤‚¨¨CCtº!ŒþGûþz§wQ$&f~moËTÛ_ÞUHˆå²ök¼ »´Ú!®=HVÖBcûìïá{+€°"uú·nؘŽ{#< š¡ü·7€m·¯à¦/t_õ£&ˆAãx<¹˜í®Þ÷¸ÿÜÝ„‚ ‚ÿ»9aÎF3à#4Ëï´Æ³ß™NÐ:ïÔCäSOóp ‰Hþ#`«¨ûäR®ÐŽúÛ¦q¾77]ÒœcÞ'¡¾|øF]•ÊÒGMå)AE‚ãL¢¢@ÂB">#4}eýP'ƒCŽ oúš‚EÏkÿ¥ð°Œ'OŒ¨­1Ìt­ªÈÏ"¶fÇâí,{RÆUTÞ”Ï=.&QFiuª|oj· 5¢¡çè•Ãp£“úphÙÞ±aåv.š¸uíÌÀÇÀ4íž#4 5*““-Ó††—ƒ,±-–Æ]Q,¡òièå›ÑQGjÕ6uiÑsYe€š4Í®TLÔ±´[Ôƃx…ºM2ÁE¡ïÉìĘY.ä‹0eÕ»&BIŽÈʦâØ kÝŠ±¿xK¦w’/>ŠU¶qß1²<d1ðÍåïk%£·Pø! Óë®c(ôLØMž§#)Çwv(WJbÉ‘ˆ—$HÉœZåçl^FÈÌ1—`Ðçí¾H6âf~9k‰eYúµcÈ ß]›Åt³­:FÆp¶ —pð#¹qKó˜¾Xͼ޳ýjÀNßUA”<ñw‘¤1'‘ÇwÔÂÇn­fªb»ÒY0NŒX¤GÆ$ðDL=>XK¸QÆp,†#<¹^>]ŸÇÉËö|6á†5â`ª¢Ò@òºñýþ~>íwë%w\Å_>{Œì3DEATLPÅJ…óÙË}Å2횀ˆ9ÂFA4Øí—#4nJgÿHÒ­–#<€ø H!.wwz<ǯ°L]Öli$ 0‰#45 ¤:<‚†ü²Zбk¨†p‘3ÅÚ@Û™‰bfpÅÿùžÙY 'ÎÆÛnLw–ç„™€ˆÄX0DåËe…ÍYšËwô­èÁ¦æ’ÛiH‰Š›‰¨¦†ÔcmK&m´[%c$i!A˜*caJª¹eB8€žüäÿlù‘J·ÌÝ$Àÿ]€iãTÝoH("‚–Å]ç‰@a.Ô ¬@O`ÓX|Ôï>žFT7·³’†oÕOÆ?8oÕ·}ÉucäÿÁ83“âm_/*Õõñ*©"(‹ñLYž½”Ÿ÷dåòÉZ.‰ž”L!Ù¶FÎKºƒ}¦•û§ú$2‰ñþÿhxqÖÛ” ñT¸±#4¯±úP=’ZóòóÀNÐxùGm: ÙÒÕÆ0Ðò‰AŸŽÚŸ!ž#4 Œ9Ñðê—r_³/m:¹¨9bÄÑë­ù<MÐf£ˆ¬ÂýÔZ5? …6ï#<lú³t­î3;†Ç¦~Öj»³×¡Èå*žl °Rj›~Vì±­å¤#<ߥ®jJÇäjºj-F7–ÛãF¯¥«|š6Ô›ƒZ¾Ù­rÕ}+×uUuîÓAL¶•ÆT¥gJÓ!Ñ®±Fe¡5EaMŒ¾ZS£pkþÒ€¯¢') 8¬ÉÖ@Úw¹µbçˆà]ìÁv¤Â&0M‹ñéÐ\IPé®ð¼Ìr¿Y|Ö,~T´p#< Nïûè¿tEÔ#4$‹ 3rnÚŠÿÙ;ü>ÉGMOEw(j}¡È_(]OàÀ(·:Ÿ)©ýÑUŒI#) #)H@„,5ºÑÄM‡v‚ö;msfµd¶Ó3\-»j»-_ƒb›5&¾¯ªÖ¿¦EFBPqG¼öv}#0óY÷O˜š¢‚>={ýéïEõû>ß“¦Æeò²Óø=“ˆ aØã&„‘OTOŒ±Aðû¿~OûÅD?.É ÛÿßÉÓJkik¾¨*H„‹‡ÿ>R ¯ûÄ©¬˜{ÿ€U#þxÿñ€$OX!ˆ —ggÿ©ÙËÿÑÿ#)¹òäË á°Yâ ÿÏÊÿ'þŸím~•ú<þÿ­ØiæÐìŽ\Ä<éwtzñã‡mø}n )ì=%…ÞÎÿOÀñïkÿ#4‡×æ:ÿï1ð¤ý1ûˆ—Ž1H@üŸò¥Ýï—ôÄ3‹!¤Lãù±aðìÍËÿ=g52Œsé{KÑXá=~_þi¯§õTÏê»_áæ%ôi¡ý‚poøT£ÿ³Ò»xœULà›éªáq–s¦ò8‡Ç×iÓ¶ˆÐeecú9ÈÓxFà±7–ÂQ¿Ñ8¨–½-§0aw<?ÿ^ŒÁÔÆ™&HG;¸ñ÷.ðþ@‘fĸ@ûü~>¤—þWïr7èiÀæ>Xž|}Ÿ÷VÖTÝG\[Õ&g¼ÃÅ´„” N@`š«^Šüßðµ=ƒéùĽßKÐDÿÅÜ‘N$/þÒ#)
+#<==
diff --git a/waflib/.gitignore b/waflib/.gitignore
new file mode 100644
index 0000000..8d35cb3
--- /dev/null
+++ b/waflib/.gitignore
@@ -0,0 +1,2 @@
+__pycache__
+*.pyc
diff --git a/Build.py b/waflib/Build.py
index 1afcba6..1afcba6 100644
--- a/Build.py
+++ b/waflib/Build.py
diff --git a/waflib/COPYING b/waflib/COPYING
new file mode 100644
index 0000000..a4147d2
--- /dev/null
+++ b/waflib/COPYING
@@ -0,0 +1,25 @@
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions
+are met:
+
+1. Redistributions of source code must retain the above copyright
+ notice, this list of conditions and the following disclaimer.
+
+2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in the
+ documentation and/or other materials provided with the distribution.
+
+3. The name of the author may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
+IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
+INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
+IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGE.
diff --git a/ConfigSet.py b/waflib/ConfigSet.py
index b300bb5..b300bb5 100644
--- a/ConfigSet.py
+++ b/waflib/ConfigSet.py
diff --git a/Configure.py b/waflib/Configure.py
index d0a4793..d0a4793 100644
--- a/Configure.py
+++ b/waflib/Configure.py
diff --git a/Context.py b/waflib/Context.py
index bb47c92..bb47c92 100644
--- a/Context.py
+++ b/waflib/Context.py
diff --git a/Errors.py b/waflib/Errors.py
index bf75c1b..bf75c1b 100644
--- a/Errors.py
+++ b/waflib/Errors.py
diff --git a/Logs.py b/waflib/Logs.py
index 2a47516..2a47516 100644
--- a/Logs.py
+++ b/waflib/Logs.py
diff --git a/Node.py b/waflib/Node.py
index 4ac1ea8..4ac1ea8 100644
--- a/Node.py
+++ b/waflib/Node.py
diff --git a/Options.py b/waflib/Options.py
index ad802d4..ad802d4 100644
--- a/Options.py
+++ b/waflib/Options.py
diff --git a/README.md b/waflib/README.md
index c5361b9..c5361b9 100644
--- a/README.md
+++ b/waflib/README.md
diff --git a/Runner.py b/waflib/Runner.py
index 261084d..261084d 100644
--- a/Runner.py
+++ b/waflib/Runner.py
diff --git a/Scripting.py b/waflib/Scripting.py
index 749d4f2..749d4f2 100644
--- a/Scripting.py
+++ b/waflib/Scripting.py
diff --git a/Task.py b/waflib/Task.py
index 0fc449d..0fc449d 100644
--- a/Task.py
+++ b/waflib/Task.py
diff --git a/TaskGen.py b/waflib/TaskGen.py
index a74e643..a74e643 100644
--- a/TaskGen.py
+++ b/waflib/TaskGen.py
diff --git a/Tools/__init__.py b/waflib/Tools/__init__.py
index 079df35..079df35 100644
--- a/Tools/__init__.py
+++ b/waflib/Tools/__init__.py
diff --git a/Tools/ar.py b/waflib/Tools/ar.py
index b39b645..b39b645 100644
--- a/Tools/ar.py
+++ b/waflib/Tools/ar.py
diff --git a/Tools/asm.py b/waflib/Tools/asm.py
index b6f26fb..b6f26fb 100644
--- a/Tools/asm.py
+++ b/waflib/Tools/asm.py
diff --git a/Tools/bison.py b/waflib/Tools/bison.py
index eef56dc..eef56dc 100644
--- a/Tools/bison.py
+++ b/waflib/Tools/bison.py
diff --git a/Tools/c.py b/waflib/Tools/c.py
index effd6b6..effd6b6 100644
--- a/Tools/c.py
+++ b/waflib/Tools/c.py
diff --git a/Tools/c_aliases.py b/waflib/Tools/c_aliases.py
index c9d5369..c9d5369 100644
--- a/Tools/c_aliases.py
+++ b/waflib/Tools/c_aliases.py
diff --git a/Tools/c_config.py b/waflib/Tools/c_config.py
index d2b3c0d..d2b3c0d 100644
--- a/Tools/c_config.py
+++ b/waflib/Tools/c_config.py
diff --git a/Tools/c_osx.py b/waflib/Tools/c_osx.py
index f70b128..f70b128 100644
--- a/Tools/c_osx.py
+++ b/waflib/Tools/c_osx.py
diff --git a/Tools/c_preproc.py b/waflib/Tools/c_preproc.py
index 7e04b4a..7e04b4a 100644
--- a/Tools/c_preproc.py
+++ b/waflib/Tools/c_preproc.py
diff --git a/Tools/c_tests.py b/waflib/Tools/c_tests.py
index f858df5..f858df5 100644
--- a/Tools/c_tests.py
+++ b/waflib/Tools/c_tests.py
diff --git a/Tools/ccroot.py b/waflib/Tools/ccroot.py
index cfef8bf..cfef8bf 100644
--- a/Tools/ccroot.py
+++ b/waflib/Tools/ccroot.py
diff --git a/Tools/clang.py b/waflib/Tools/clang.py
index 3828e39..3828e39 100644
--- a/Tools/clang.py
+++ b/waflib/Tools/clang.py
diff --git a/Tools/clangxx.py b/waflib/Tools/clangxx.py
index 152013c..152013c 100644
--- a/Tools/clangxx.py
+++ b/waflib/Tools/clangxx.py
diff --git a/Tools/compiler_c.py b/waflib/Tools/compiler_c.py
index 2dba3f8..2dba3f8 100644
--- a/Tools/compiler_c.py
+++ b/waflib/Tools/compiler_c.py
diff --git a/Tools/compiler_cxx.py b/waflib/Tools/compiler_cxx.py
index 1af65a2..1af65a2 100644
--- a/Tools/compiler_cxx.py
+++ b/waflib/Tools/compiler_cxx.py
diff --git a/Tools/compiler_d.py b/waflib/Tools/compiler_d.py
index 43bb1f6..43bb1f6 100644
--- a/Tools/compiler_d.py
+++ b/waflib/Tools/compiler_d.py
diff --git a/Tools/compiler_fc.py b/waflib/Tools/compiler_fc.py
index 96b58e7..96b58e7 100644
--- a/Tools/compiler_fc.py
+++ b/waflib/Tools/compiler_fc.py
diff --git a/Tools/cs.py b/waflib/Tools/cs.py
index aecca6d..aecca6d 100644
--- a/Tools/cs.py
+++ b/waflib/Tools/cs.py
diff --git a/Tools/cxx.py b/waflib/Tools/cxx.py
index 194fad7..194fad7 100644
--- a/Tools/cxx.py
+++ b/waflib/Tools/cxx.py
diff --git a/Tools/d.py b/waflib/Tools/d.py
index e4cf73b..e4cf73b 100644
--- a/Tools/d.py
+++ b/waflib/Tools/d.py
diff --git a/Tools/d_config.py b/waflib/Tools/d_config.py
index 6637556..6637556 100644
--- a/Tools/d_config.py
+++ b/waflib/Tools/d_config.py
diff --git a/Tools/d_scan.py b/waflib/Tools/d_scan.py
index 14c6c31..14c6c31 100644
--- a/Tools/d_scan.py
+++ b/waflib/Tools/d_scan.py
diff --git a/Tools/dbus.py b/waflib/Tools/dbus.py
index d520f1c..d520f1c 100644
--- a/Tools/dbus.py
+++ b/waflib/Tools/dbus.py
diff --git a/Tools/dmd.py b/waflib/Tools/dmd.py
index 8917ca1..8917ca1 100644
--- a/Tools/dmd.py
+++ b/waflib/Tools/dmd.py
diff --git a/Tools/errcheck.py b/waflib/Tools/errcheck.py
index de8d75a..de8d75a 100644
--- a/Tools/errcheck.py
+++ b/waflib/Tools/errcheck.py
diff --git a/Tools/fc.py b/waflib/Tools/fc.py
index d9e8d8c..d9e8d8c 100644
--- a/Tools/fc.py
+++ b/waflib/Tools/fc.py
diff --git a/Tools/fc_config.py b/waflib/Tools/fc_config.py
index 222f3a5..222f3a5 100644
--- a/Tools/fc_config.py
+++ b/waflib/Tools/fc_config.py
diff --git a/Tools/fc_scan.py b/waflib/Tools/fc_scan.py
index 12cb0fc..12cb0fc 100644
--- a/Tools/fc_scan.py
+++ b/waflib/Tools/fc_scan.py
diff --git a/Tools/flex.py b/waflib/Tools/flex.py
index 2256657..2256657 100644
--- a/Tools/flex.py
+++ b/waflib/Tools/flex.py
diff --git a/Tools/g95.py b/waflib/Tools/g95.py
index f69ba4f..f69ba4f 100644
--- a/Tools/g95.py
+++ b/waflib/Tools/g95.py
diff --git a/Tools/gas.py b/waflib/Tools/gas.py
index 77afed7..77afed7 100644
--- a/Tools/gas.py
+++ b/waflib/Tools/gas.py
diff --git a/Tools/gcc.py b/waflib/Tools/gcc.py
index acdd473..acdd473 100644
--- a/Tools/gcc.py
+++ b/waflib/Tools/gcc.py
diff --git a/Tools/gdc.py b/waflib/Tools/gdc.py
index d89a66d..d89a66d 100644
--- a/Tools/gdc.py
+++ b/waflib/Tools/gdc.py
diff --git a/Tools/gfortran.py b/waflib/Tools/gfortran.py
index 1050667..1050667 100644
--- a/Tools/gfortran.py
+++ b/waflib/Tools/gfortran.py
diff --git a/Tools/glib2.py b/waflib/Tools/glib2.py
index 949fe37..949fe37 100644
--- a/Tools/glib2.py
+++ b/waflib/Tools/glib2.py
diff --git a/Tools/gnu_dirs.py b/waflib/Tools/gnu_dirs.py
index 2847071..2847071 100644
--- a/Tools/gnu_dirs.py
+++ b/waflib/Tools/gnu_dirs.py
diff --git a/Tools/gxx.py b/waflib/Tools/gxx.py
index 22c5d26..22c5d26 100644
--- a/Tools/gxx.py
+++ b/waflib/Tools/gxx.py
diff --git a/Tools/icc.py b/waflib/Tools/icc.py
index b6492c8..b6492c8 100644
--- a/Tools/icc.py
+++ b/waflib/Tools/icc.py
diff --git a/Tools/icpc.py b/waflib/Tools/icpc.py
index 8a6cc6c..8a6cc6c 100644
--- a/Tools/icpc.py
+++ b/waflib/Tools/icpc.py
diff --git a/Tools/ifort.py b/waflib/Tools/ifort.py
index 74934f3..74934f3 100644
--- a/Tools/ifort.py
+++ b/waflib/Tools/ifort.py
diff --git a/Tools/intltool.py b/waflib/Tools/intltool.py
index af95ba8..af95ba8 100644
--- a/Tools/intltool.py
+++ b/waflib/Tools/intltool.py
diff --git a/Tools/irixcc.py b/waflib/Tools/irixcc.py
index c3ae1ac..c3ae1ac 100644
--- a/Tools/irixcc.py
+++ b/waflib/Tools/irixcc.py
diff --git a/Tools/javaw.py b/waflib/Tools/javaw.py
index f6fd20c..f6fd20c 100644
--- a/Tools/javaw.py
+++ b/waflib/Tools/javaw.py
diff --git a/Tools/ldc2.py b/waflib/Tools/ldc2.py
index a51c344..a51c344 100644
--- a/Tools/ldc2.py
+++ b/waflib/Tools/ldc2.py
diff --git a/Tools/lua.py b/waflib/Tools/lua.py
index 15a333a..15a333a 100644
--- a/Tools/lua.py
+++ b/waflib/Tools/lua.py
diff --git a/Tools/md5_tstamp.py b/waflib/Tools/md5_tstamp.py
index 6428e46..6428e46 100644
--- a/Tools/md5_tstamp.py
+++ b/waflib/Tools/md5_tstamp.py
diff --git a/Tools/msvc.py b/waflib/Tools/msvc.py
index 17b347d..17b347d 100644
--- a/Tools/msvc.py
+++ b/waflib/Tools/msvc.py
diff --git a/Tools/nasm.py b/waflib/Tools/nasm.py
index 411d582..411d582 100644
--- a/Tools/nasm.py
+++ b/waflib/Tools/nasm.py
diff --git a/Tools/nobuild.py b/waflib/Tools/nobuild.py
index 2e4b055..2e4b055 100644
--- a/Tools/nobuild.py
+++ b/waflib/Tools/nobuild.py
diff --git a/Tools/perl.py b/waflib/Tools/perl.py
index 32b03fb..32b03fb 100644
--- a/Tools/perl.py
+++ b/waflib/Tools/perl.py
diff --git a/Tools/python.py b/waflib/Tools/python.py
index 25841d0..25841d0 100644
--- a/Tools/python.py
+++ b/waflib/Tools/python.py
diff --git a/Tools/qt5.py b/waflib/Tools/qt5.py
index 4f9c690..4f9c690 100644
--- a/Tools/qt5.py
+++ b/waflib/Tools/qt5.py
diff --git a/Tools/ruby.py b/waflib/Tools/ruby.py
index 8d92a79..8d92a79 100644
--- a/Tools/ruby.py
+++ b/waflib/Tools/ruby.py
diff --git a/Tools/suncc.py b/waflib/Tools/suncc.py
index 33d34fc..33d34fc 100644
--- a/Tools/suncc.py
+++ b/waflib/Tools/suncc.py
diff --git a/Tools/suncxx.py b/waflib/Tools/suncxx.py
index 3b384f6..3b384f6 100644
--- a/Tools/suncxx.py
+++ b/waflib/Tools/suncxx.py
diff --git a/Tools/tex.py b/waflib/Tools/tex.py
index eaf9fdb..eaf9fdb 100644
--- a/Tools/tex.py
+++ b/waflib/Tools/tex.py
diff --git a/Tools/vala.py b/waflib/Tools/vala.py
index 822ec50..822ec50 100644
--- a/Tools/vala.py
+++ b/waflib/Tools/vala.py
diff --git a/Tools/waf_unit_test.py b/waflib/Tools/waf_unit_test.py
index a71ed1c..a71ed1c 100644
--- a/Tools/waf_unit_test.py
+++ b/waflib/Tools/waf_unit_test.py
diff --git a/Tools/winres.py b/waflib/Tools/winres.py
index 586c596..586c596 100644
--- a/Tools/winres.py
+++ b/waflib/Tools/winres.py
diff --git a/Tools/xlc.py b/waflib/Tools/xlc.py
index 134dd41..134dd41 100644
--- a/Tools/xlc.py
+++ b/waflib/Tools/xlc.py
diff --git a/Tools/xlcxx.py b/waflib/Tools/xlcxx.py
index 76aa59b..76aa59b 100644
--- a/Tools/xlcxx.py
+++ b/waflib/Tools/xlcxx.py
diff --git a/Utils.py b/waflib/Utils.py
index b4665c4..b4665c4 100644
--- a/Utils.py
+++ b/waflib/Utils.py
diff --git a/__init__.py b/waflib/__init__.py
index 079df35..079df35 100644
--- a/__init__.py
+++ b/waflib/__init__.py
diff --git a/ansiterm.py b/waflib/ansiterm.py
index 0d20c63..0d20c63 100644
--- a/ansiterm.py
+++ b/waflib/ansiterm.py
diff --git a/extras/__init__.py b/waflib/extras/__init__.py
index c8a3c34..c8a3c34 100644
--- a/extras/__init__.py
+++ b/waflib/extras/__init__.py
diff --git a/extras/autowaf.py b/waflib/extras/autowaf.py
index a4a06ba..a4a06ba 100644
--- a/extras/autowaf.py
+++ b/waflib/extras/autowaf.py
diff --git a/extras/batched_cc.py b/waflib/extras/batched_cc.py
index aad2872..aad2872 100644
--- a/extras/batched_cc.py
+++ b/waflib/extras/batched_cc.py
diff --git a/extras/biber.py b/waflib/extras/biber.py
index fd9db4e..fd9db4e 100644
--- a/extras/biber.py
+++ b/waflib/extras/biber.py
diff --git a/extras/bjam.py b/waflib/extras/bjam.py
index 8e04d3a..8e04d3a 100644
--- a/extras/bjam.py
+++ b/waflib/extras/bjam.py
diff --git a/extras/blender.py b/waflib/extras/blender.py
index e5efc28..e5efc28 100644
--- a/extras/blender.py
+++ b/waflib/extras/blender.py
diff --git a/extras/boo.py b/waflib/extras/boo.py
index 06623d4..06623d4 100644
--- a/extras/boo.py
+++ b/waflib/extras/boo.py
diff --git a/extras/boost.py b/waflib/extras/boost.py
index c2aaaa9..c2aaaa9 100644
--- a/extras/boost.py
+++ b/waflib/extras/boost.py
diff --git a/extras/build_file_tracker.py b/waflib/extras/build_file_tracker.py
index c4f26fd..c4f26fd 100644
--- a/extras/build_file_tracker.py
+++ b/waflib/extras/build_file_tracker.py
diff --git a/extras/build_logs.py b/waflib/extras/build_logs.py
index cdf8ed0..cdf8ed0 100644
--- a/extras/build_logs.py
+++ b/waflib/extras/build_logs.py
diff --git a/extras/buildcopy.py b/waflib/extras/buildcopy.py
index a6d9ac8..a6d9ac8 100644
--- a/extras/buildcopy.py
+++ b/waflib/extras/buildcopy.py
diff --git a/extras/c_bgxlc.py b/waflib/extras/c_bgxlc.py
index 6e3eaf7..6e3eaf7 100644
--- a/extras/c_bgxlc.py
+++ b/waflib/extras/c_bgxlc.py
diff --git a/extras/c_dumbpreproc.py b/waflib/extras/c_dumbpreproc.py
index ce9e1a4..ce9e1a4 100644
--- a/extras/c_dumbpreproc.py
+++ b/waflib/extras/c_dumbpreproc.py
diff --git a/extras/c_emscripten.py b/waflib/extras/c_emscripten.py
index e1ac494..e1ac494 100644
--- a/extras/c_emscripten.py
+++ b/waflib/extras/c_emscripten.py
diff --git a/extras/c_nec.py b/waflib/extras/c_nec.py
index 96bfae4..96bfae4 100644
--- a/extras/c_nec.py
+++ b/waflib/extras/c_nec.py
diff --git a/extras/cabal.py b/waflib/extras/cabal.py
index e10a0d1..e10a0d1 100644
--- a/extras/cabal.py
+++ b/waflib/extras/cabal.py
diff --git a/extras/cfg_altoptions.py b/waflib/extras/cfg_altoptions.py
index 47b1189..47b1189 100644
--- a/extras/cfg_altoptions.py
+++ b/waflib/extras/cfg_altoptions.py
diff --git a/extras/clang_compilation_database.py b/waflib/extras/clang_compilation_database.py
index 4d9b5e2..4d9b5e2 100644
--- a/extras/clang_compilation_database.py
+++ b/waflib/extras/clang_compilation_database.py
diff --git a/extras/codelite.py b/waflib/extras/codelite.py
index 523302c..523302c 100644
--- a/extras/codelite.py
+++ b/waflib/extras/codelite.py
diff --git a/extras/color_gcc.py b/waflib/extras/color_gcc.py
index b68c5eb..b68c5eb 100644
--- a/extras/color_gcc.py
+++ b/waflib/extras/color_gcc.py
diff --git a/extras/color_rvct.py b/waflib/extras/color_rvct.py
index f89ccbd..f89ccbd 100644
--- a/extras/color_rvct.py
+++ b/waflib/extras/color_rvct.py
diff --git a/extras/compat15.py b/waflib/extras/compat15.py
index 0e74df8..0e74df8 100644
--- a/extras/compat15.py
+++ b/waflib/extras/compat15.py
diff --git a/extras/cppcheck.py b/waflib/extras/cppcheck.py
index 13ff424..13ff424 100644
--- a/extras/cppcheck.py
+++ b/waflib/extras/cppcheck.py
diff --git a/extras/cpplint.py b/waflib/extras/cpplint.py
index e3302e5..e3302e5 100644
--- a/extras/cpplint.py
+++ b/waflib/extras/cpplint.py
diff --git a/extras/cross_gnu.py b/waflib/extras/cross_gnu.py
index 309f53b..309f53b 100644
--- a/extras/cross_gnu.py
+++ b/waflib/extras/cross_gnu.py
diff --git a/extras/cython.py b/waflib/extras/cython.py
index 481d6f4..481d6f4 100644
--- a/extras/cython.py
+++ b/waflib/extras/cython.py
diff --git a/extras/dcc.py b/waflib/extras/dcc.py
index c1a57c0..c1a57c0 100644
--- a/extras/dcc.py
+++ b/waflib/extras/dcc.py
diff --git a/extras/distnet.py b/waflib/extras/distnet.py
index 09a31a6..09a31a6 100644
--- a/extras/distnet.py
+++ b/waflib/extras/distnet.py
diff --git a/extras/doxygen.py b/waflib/extras/doxygen.py
index 28f56e9..28f56e9 100644
--- a/extras/doxygen.py
+++ b/waflib/extras/doxygen.py
diff --git a/extras/dpapi.py b/waflib/extras/dpapi.py
index b94d482..b94d482 100644
--- a/extras/dpapi.py
+++ b/waflib/extras/dpapi.py
diff --git a/extras/eclipse.py b/waflib/extras/eclipse.py
index bb78741..bb78741 100644
--- a/extras/eclipse.py
+++ b/waflib/extras/eclipse.py
diff --git a/extras/erlang.py b/waflib/extras/erlang.py
index 49f6d5b..49f6d5b 100644
--- a/extras/erlang.py
+++ b/waflib/extras/erlang.py
diff --git a/extras/fast_partial.py b/waflib/extras/fast_partial.py
index b3af513..b3af513 100644
--- a/extras/fast_partial.py
+++ b/waflib/extras/fast_partial.py
diff --git a/extras/fc_bgxlf.py b/waflib/extras/fc_bgxlf.py
index cca1810..cca1810 100644
--- a/extras/fc_bgxlf.py
+++ b/waflib/extras/fc_bgxlf.py
diff --git a/extras/fc_cray.py b/waflib/extras/fc_cray.py
index da733fa..da733fa 100644
--- a/extras/fc_cray.py
+++ b/waflib/extras/fc_cray.py
diff --git a/extras/fc_nag.py b/waflib/extras/fc_nag.py
index edcb218..edcb218 100644
--- a/extras/fc_nag.py
+++ b/waflib/extras/fc_nag.py
diff --git a/extras/fc_nec.py b/waflib/extras/fc_nec.py
index 67c8680..67c8680 100644
--- a/extras/fc_nec.py
+++ b/waflib/extras/fc_nec.py
diff --git a/extras/fc_open64.py b/waflib/extras/fc_open64.py
index 413719f..413719f 100644
--- a/extras/fc_open64.py
+++ b/waflib/extras/fc_open64.py
diff --git a/extras/fc_pgfortran.py b/waflib/extras/fc_pgfortran.py
index afb2817..afb2817 100644
--- a/extras/fc_pgfortran.py
+++ b/waflib/extras/fc_pgfortran.py
diff --git a/extras/fc_solstudio.py b/waflib/extras/fc_solstudio.py
index 53766df..53766df 100644
--- a/extras/fc_solstudio.py
+++ b/waflib/extras/fc_solstudio.py
diff --git a/extras/fc_xlf.py b/waflib/extras/fc_xlf.py
index 5a3da03..5a3da03 100644
--- a/extras/fc_xlf.py
+++ b/waflib/extras/fc_xlf.py
diff --git a/extras/file_to_object.py b/waflib/extras/file_to_object.py
index 1393b51..1393b51 100644
--- a/extras/file_to_object.py
+++ b/waflib/extras/file_to_object.py
diff --git a/extras/fluid.py b/waflib/extras/fluid.py
index 4814a35..4814a35 100644
--- a/extras/fluid.py
+++ b/waflib/extras/fluid.py
diff --git a/extras/freeimage.py b/waflib/extras/freeimage.py
index f27e525..f27e525 100644
--- a/extras/freeimage.py
+++ b/waflib/extras/freeimage.py
diff --git a/extras/fsb.py b/waflib/extras/fsb.py
index 1b8f398..1b8f398 100644
--- a/extras/fsb.py
+++ b/waflib/extras/fsb.py
diff --git a/extras/fsc.py b/waflib/extras/fsc.py
index c67e70b..c67e70b 100644
--- a/extras/fsc.py
+++ b/waflib/extras/fsc.py
diff --git a/extras/gccdeps.py b/waflib/extras/gccdeps.py
index d9758ab..d9758ab 100644
--- a/extras/gccdeps.py
+++ b/waflib/extras/gccdeps.py
diff --git a/extras/gdbus.py b/waflib/extras/gdbus.py
index 0e0476e..0e0476e 100644
--- a/extras/gdbus.py
+++ b/waflib/extras/gdbus.py
diff --git a/extras/gob2.py b/waflib/extras/gob2.py
index b4fa3b9..b4fa3b9 100644
--- a/extras/gob2.py
+++ b/waflib/extras/gob2.py
diff --git a/extras/halide.py b/waflib/extras/halide.py
index 6078e38..6078e38 100644
--- a/extras/halide.py
+++ b/waflib/extras/halide.py
diff --git a/extras/javatest.py b/waflib/extras/javatest.py
index 979b8d8..979b8d8 100755
--- a/extras/javatest.py
+++ b/waflib/extras/javatest.py
diff --git a/extras/kde4.py b/waflib/extras/kde4.py
index e49a9ec..e49a9ec 100644
--- a/extras/kde4.py
+++ b/waflib/extras/kde4.py
diff --git a/extras/local_rpath.py b/waflib/extras/local_rpath.py
index b2507e1..b2507e1 100644
--- a/extras/local_rpath.py
+++ b/waflib/extras/local_rpath.py
diff --git a/extras/lv2.py b/waflib/extras/lv2.py
index 815987f..815987f 100644
--- a/extras/lv2.py
+++ b/waflib/extras/lv2.py
diff --git a/extras/make.py b/waflib/extras/make.py
index 933d9ca..933d9ca 100644
--- a/extras/make.py
+++ b/waflib/extras/make.py
diff --git a/extras/midl.py b/waflib/extras/midl.py
index 43e6cf9..43e6cf9 100644
--- a/extras/midl.py
+++ b/waflib/extras/midl.py
diff --git a/extras/msvcdeps.py b/waflib/extras/msvcdeps.py
index fc1ecd4..fc1ecd4 100644
--- a/extras/msvcdeps.py
+++ b/waflib/extras/msvcdeps.py
diff --git a/extras/msvs.py b/waflib/extras/msvs.py
index 8aa2db0..8aa2db0 100644
--- a/extras/msvs.py
+++ b/waflib/extras/msvs.py
diff --git a/extras/netcache_client.py b/waflib/extras/netcache_client.py
index dc49048..dc49048 100644
--- a/extras/netcache_client.py
+++ b/waflib/extras/netcache_client.py
diff --git a/extras/objcopy.py b/waflib/extras/objcopy.py
index 82d8359..82d8359 100644
--- a/extras/objcopy.py
+++ b/waflib/extras/objcopy.py
diff --git a/extras/ocaml.py b/waflib/extras/ocaml.py
index afe73c0..afe73c0 100644
--- a/extras/ocaml.py
+++ b/waflib/extras/ocaml.py
diff --git a/extras/package.py b/waflib/extras/package.py
index c06498e..c06498e 100644
--- a/extras/package.py
+++ b/waflib/extras/package.py
diff --git a/extras/parallel_debug.py b/waflib/extras/parallel_debug.py
index 35883a3..35883a3 100644
--- a/extras/parallel_debug.py
+++ b/waflib/extras/parallel_debug.py
diff --git a/extras/pch.py b/waflib/extras/pch.py
index 103e752..103e752 100644
--- a/extras/pch.py
+++ b/waflib/extras/pch.py
diff --git a/extras/pep8.py b/waflib/extras/pep8.py
index 676beed..676beed 100644
--- a/extras/pep8.py
+++ b/waflib/extras/pep8.py
diff --git a/extras/pgicc.py b/waflib/extras/pgicc.py
index 9790b9c..9790b9c 100644
--- a/extras/pgicc.py
+++ b/waflib/extras/pgicc.py
diff --git a/extras/pgicxx.py b/waflib/extras/pgicxx.py
index eae121c..eae121c 100644
--- a/extras/pgicxx.py
+++ b/waflib/extras/pgicxx.py
diff --git a/extras/proc.py b/waflib/extras/proc.py
index 764abec..764abec 100644
--- a/extras/proc.py
+++ b/waflib/extras/proc.py
diff --git a/extras/protoc.py b/waflib/extras/protoc.py
index f3cb4d8..f3cb4d8 100644
--- a/extras/protoc.py
+++ b/waflib/extras/protoc.py
diff --git a/extras/pyqt5.py b/waflib/extras/pyqt5.py
index c21dfa7..c21dfa7 100644
--- a/extras/pyqt5.py
+++ b/waflib/extras/pyqt5.py
diff --git a/extras/pytest.py b/waflib/extras/pytest.py
index 7dd5a1a..7dd5a1a 100644
--- a/extras/pytest.py
+++ b/waflib/extras/pytest.py
diff --git a/extras/qnxnto.py b/waflib/extras/qnxnto.py
index 1158124..1158124 100644
--- a/extras/qnxnto.py
+++ b/waflib/extras/qnxnto.py
diff --git a/extras/qt4.py b/waflib/extras/qt4.py
index 90cae7e..90cae7e 100644
--- a/extras/qt4.py
+++ b/waflib/extras/qt4.py
diff --git a/extras/relocation.py b/waflib/extras/relocation.py
index 7e821f4..7e821f4 100644
--- a/extras/relocation.py
+++ b/waflib/extras/relocation.py
diff --git a/extras/remote.py b/waflib/extras/remote.py
index 3b038f7..3b038f7 100644
--- a/extras/remote.py
+++ b/waflib/extras/remote.py
diff --git a/extras/resx.py b/waflib/extras/resx.py
index caf4d31..caf4d31 100644
--- a/extras/resx.py
+++ b/waflib/extras/resx.py
diff --git a/extras/review.py b/waflib/extras/review.py
index 561e062..561e062 100644
--- a/extras/review.py
+++ b/waflib/extras/review.py
diff --git a/extras/rst.py b/waflib/extras/rst.py
index f3c3a5e..f3c3a5e 100644
--- a/extras/rst.py
+++ b/waflib/extras/rst.py
diff --git a/extras/run_do_script.py b/waflib/extras/run_do_script.py
index f3c5812..f3c5812 100644
--- a/extras/run_do_script.py
+++ b/waflib/extras/run_do_script.py
diff --git a/extras/run_m_script.py b/waflib/extras/run_m_script.py
index b5f27eb..b5f27eb 100644
--- a/extras/run_m_script.py
+++ b/waflib/extras/run_m_script.py
diff --git a/extras/run_py_script.py b/waflib/extras/run_py_script.py
index 3670381..3670381 100644
--- a/extras/run_py_script.py
+++ b/waflib/extras/run_py_script.py
diff --git a/extras/run_r_script.py b/waflib/extras/run_r_script.py
index b0d8f2b..b0d8f2b 100644
--- a/extras/run_r_script.py
+++ b/waflib/extras/run_r_script.py
diff --git a/extras/sas.py b/waflib/extras/sas.py
index 754c614..754c614 100644
--- a/extras/sas.py
+++ b/waflib/extras/sas.py
diff --git a/extras/satellite_assembly.py b/waflib/extras/satellite_assembly.py
index 005eb07..005eb07 100644
--- a/extras/satellite_assembly.py
+++ b/waflib/extras/satellite_assembly.py
diff --git a/extras/scala.py b/waflib/extras/scala.py
index a9880f0..a9880f0 100644
--- a/extras/scala.py
+++ b/waflib/extras/scala.py
diff --git a/extras/slow_qt4.py b/waflib/extras/slow_qt4.py
index ec7880b..ec7880b 100644
--- a/extras/slow_qt4.py
+++ b/waflib/extras/slow_qt4.py
diff --git a/extras/softlink_libs.py b/waflib/extras/softlink_libs.py
index 50c777f..50c777f 100644
--- a/extras/softlink_libs.py
+++ b/waflib/extras/softlink_libs.py
diff --git a/extras/stale.py b/waflib/extras/stale.py
index cac3f46..cac3f46 100644
--- a/extras/stale.py
+++ b/waflib/extras/stale.py
diff --git a/extras/stracedeps.py b/waflib/extras/stracedeps.py
index 37d82cb..37d82cb 100644
--- a/extras/stracedeps.py
+++ b/waflib/extras/stracedeps.py
diff --git a/extras/swig.py b/waflib/extras/swig.py
index fd3d6d2..fd3d6d2 100644
--- a/extras/swig.py
+++ b/waflib/extras/swig.py
diff --git a/extras/syms.py b/waflib/extras/syms.py
index dfa0059..dfa0059 100644
--- a/extras/syms.py
+++ b/waflib/extras/syms.py
diff --git a/extras/ticgt.py b/waflib/extras/ticgt.py
index f43a7ea..f43a7ea 100644
--- a/extras/ticgt.py
+++ b/waflib/extras/ticgt.py
diff --git a/extras/unity.py b/waflib/extras/unity.py
index 78128ed..78128ed 100644
--- a/extras/unity.py
+++ b/waflib/extras/unity.py
diff --git a/extras/use_config.py b/waflib/extras/use_config.py
index ef5129f..ef5129f 100644
--- a/extras/use_config.py
+++ b/waflib/extras/use_config.py
diff --git a/extras/valadoc.py b/waflib/extras/valadoc.py
index c50f69e..c50f69e 100644
--- a/extras/valadoc.py
+++ b/waflib/extras/valadoc.py
diff --git a/extras/waf_xattr.py b/waflib/extras/waf_xattr.py
index 351dd63..351dd63 100644
--- a/extras/waf_xattr.py
+++ b/waflib/extras/waf_xattr.py
diff --git a/extras/why.py b/waflib/extras/why.py
index 1bb941f..1bb941f 100644
--- a/extras/why.py
+++ b/waflib/extras/why.py
diff --git a/extras/win32_opts.py b/waflib/extras/win32_opts.py
index 9f7443c..9f7443c 100644
--- a/extras/win32_opts.py
+++ b/waflib/extras/win32_opts.py
diff --git a/extras/wix.py b/waflib/extras/wix.py
index d87bfbb..d87bfbb 100644
--- a/extras/wix.py
+++ b/waflib/extras/wix.py
diff --git a/extras/xcode6.py b/waflib/extras/xcode6.py
index 91bbff1..91bbff1 100644
--- a/extras/xcode6.py
+++ b/waflib/extras/xcode6.py
diff --git a/fixpy2.py b/waflib/fixpy2.py
index 24176e0..24176e0 100644
--- a/fixpy2.py
+++ b/waflib/fixpy2.py
diff --git a/processor.py b/waflib/processor.py
index 2eecf3b..2eecf3b 100755
--- a/processor.py
+++ b/waflib/processor.py
diff --git a/waflib/waf b/waflib/waf
new file mode 100755
index 0000000..e22930a
--- /dev/null
+++ b/waflib/waf
@@ -0,0 +1,16 @@
+#!/usr/bin/env python
+
+# Minimal waf script for projects that include waflib directly
+
+from waflib import Context, Scripting
+
+import inspect
+import os
+
+def main():
+ script_path = os.path.abspath(inspect.getfile(inspect.getmodule(main)))
+ project_path = os.path.dirname(script_path)
+ Scripting.waf_entry_point(os.getcwd(), Context.WAFVERSION, project_path)
+
+if __name__ == '__main__':
+ main()
diff --git a/wscript b/wscript
new file mode 100644
index 0000000..6ca76f7
--- /dev/null
+++ b/wscript
@@ -0,0 +1,422 @@
+#!/usr/bin/env python
+import os
+import subprocess
+import waflib.TaskGen as TaskGen
+import waflib.extras.autowaf as autowaf
+
+# Semver package/library version
+SUIL_VERSION = '0.10.0'
+SUIL_MAJOR_VERSION = SUIL_VERSION[0:SUIL_VERSION.find('.')]
+
+# Mandatory waf variables
+APPNAME = 'suil' # Package name for waf dist
+VERSION = SUIL_VERSION # Package version for waf dist
+top = '.' # Source directory
+out = 'build' # Build directory
+
+def options(ctx):
+ ctx.load('compiler_c')
+ ctx.load('compiler_cxx')
+ autowaf.set_options(ctx)
+ opt = ctx.get_option_group('Configuration options')
+ opt.add_option('--static', action='store_true', dest='static',
+ help='build static library')
+ opt.add_option('--no-shared', action='store_true', dest='no_shared',
+ help='do not build shared library')
+ opt.add_option('--no-gtk', action='store_true', dest='no_gtk',
+ help='do not build support for Gtk')
+ opt.add_option('--no-qt', action='store_true', dest='no_qt',
+ help='do not build support for Qt (any version)')
+ opt.add_option('--no-qt4', action='store_true', dest='no_qt4',
+ help='do not build support for Qt4')
+ opt.add_option('--no-qt5', action='store_true', dest='no_qt5',
+ help='do not build support for Qt5')
+ opt.add_option('--gtk2-lib-name', type='string', dest='gtk2_lib_name',
+ default="libgtk-x11-2.0.so.0",
+ help="Gtk2 library name [Default: libgtk-x11-2.0.so.0]")
+ opt.add_option('--gtk3-lib-name', type='string', dest='gtk3_lib_name',
+ default="libgtk-x11-3.0.so.0",
+ help="Gtk3 library name [Default: libgtk-x11-3.0.so.0]")
+
+def configure(conf):
+ autowaf.display_header('Suil Configuration')
+ autowaf.set_line_just(conf, 42)
+ conf.load('compiler_c', cache=True)
+ conf.load('compiler_cxx', cache=True)
+ conf.load('autowaf', cache=True)
+ autowaf.set_c_lang(conf, 'c99')
+
+ conf.env.BUILD_SHARED = not conf.options.no_shared
+ conf.env.BUILD_STATIC = conf.options.static
+
+ if not conf.env.BUILD_SHARED and not conf.env.BUILD_STATIC:
+ conf.fatal('Neither a shared nor a static build requested')
+
+ conf.env.NODELETE_FLAGS = []
+ if (not conf.env.MSVC_COMPILER and
+ conf.check(linkflags = ['-Wl,-z,nodelete'],
+ msg = 'Checking for link flags -Wl,-z,-nodelete',
+ mandatory = False)):
+ conf.env.NODELETE_FLAGS = ['-Wl,-z,nodelete']
+
+ autowaf.check_pkg(conf, 'lv2', atleast_version='1.12.0', uselib_store='LV2')
+ autowaf.check_pkg(conf, 'x11', uselib_store='X11', mandatory=False)
+
+ if not conf.options.no_gtk:
+ autowaf.check_pkg(conf, 'gtk+-2.0', uselib_store='GTK2',
+ atleast_version='2.18.0', mandatory=False)
+ if not conf.env.HAVE_GTK2:
+ autowaf.check_pkg(conf, 'gtk+-2.0', uselib_store='GTK2',
+ atleast_version='2.0.0', mandatory=False)
+ if conf.env.HAVE_GTK2:
+ autowaf.define(conf, 'SUIL_OLD_GTK', 1)
+
+ autowaf.check_pkg(conf, 'gtk+-x11-2.0', uselib_store='GTK2_X11',
+ atleast_version='2.0.0', mandatory=False)
+
+ autowaf.check_pkg(conf, 'gtk+-quartz-2.0', uselib_store='GTK2_QUARTZ',
+ atleast_version='2.0.0', mandatory=False)
+
+ autowaf.check_pkg(conf, 'gtk+-3.0', uselib_store='GTK3',
+ atleast_version='3.14.0', mandatory=False)
+
+ autowaf.check_pkg(conf, 'gtk+-x11-3.0', uselib_store='GTK3_X11',
+ atleast_version='3.14.0', mandatory=False)
+
+ if not conf.options.no_qt:
+ if not conf.options.no_qt4:
+ autowaf.check_pkg(conf, 'QtGui', uselib_store='QT4',
+ atleast_version='4.4.0', mandatory=False)
+
+ if not conf.options.no_qt5:
+ autowaf.check_pkg(conf, 'Qt5Widgets', uselib_store='QT5',
+ atleast_version='5.1.0', mandatory=False)
+ if conf.check_cxx(header_name = 'QMacCocoaViewContainer',
+ uselib = 'QT5',
+ mandatory = False):
+ autowaf.define(conf, 'SUIL_WITH_COCOA_IN_QT5', 1)
+
+ conf.check_cc(define_name = 'HAVE_LIBDL',
+ lib = 'dl',
+ mandatory = False)
+
+ autowaf.define(conf, 'SUIL_VERSION', SUIL_VERSION)
+ autowaf.define(conf, 'SUIL_MODULE_DIR',
+ conf.env.LIBDIR + '/suil-' + SUIL_MAJOR_VERSION)
+ autowaf.define(conf, 'SUIL_DIR_SEP', '/')
+ autowaf.define(conf, 'SUIL_GTK2_LIB_NAME', conf.options.gtk2_lib_name);
+ autowaf.define(conf, 'SUIL_GTK3_LIB_NAME', conf.options.gtk3_lib_name);
+
+ if conf.env.HAVE_GTK2 and conf.env.HAVE_QT4:
+ autowaf.define(conf, 'SUIL_WITH_QT4_IN_GTK2', 1)
+ if conf.env.HAVE_GTK2_X11:
+ autowaf.define(conf, 'SUIL_WITH_GTK2_IN_QT4', 1)
+
+ if conf.env.HAVE_GTK2 and conf.env.HAVE_QT5:
+ autowaf.define(conf, 'SUIL_WITH_GTK2_IN_QT5', 1)
+ autowaf.define(conf, 'SUIL_WITH_QT5_IN_GTK2', 1)
+
+ if conf.env.HAVE_GTK2 and conf.env.HAVE_GTK2_X11:
+ autowaf.define(conf, 'SUIL_WITH_X11_IN_GTK2', 1)
+
+ if conf.env.HAVE_GTK3 and conf.env.HAVE_GTK3_X11:
+ autowaf.define(conf, 'SUIL_WITH_X11_IN_GTK3', 1)
+
+ if conf.env.HAVE_GTK2 and conf.env.HAVE_GTK2_QUARTZ:
+ autowaf.define(conf, 'SUIL_WITH_COCOA_IN_GTK2', 1)
+
+ if conf.env.HAVE_GTK2 and conf.env.DEST_OS == 'win32':
+ autowaf.define(conf, 'SUIL_WITH_WIN_IN_GTK2', 1)
+
+ if conf.env.HAVE_QT4:
+ autowaf.define(conf, 'SUIL_WITH_X11_IN_QT4', 1)
+
+ if conf.env.HAVE_QT5:
+ autowaf.define(conf, 'SUIL_WITH_X11_IN_QT5', 1)
+
+ if conf.env.HAVE_X11:
+ autowaf.define(conf, 'SUIL_WITH_X11', 1)
+
+ module_prefix = ''
+ module_ext = ''
+ if conf.env.PARDEBUG:
+ module_ext += 'D'
+ if conf.env.DEST_OS == 'win32':
+ module_ext += '.dll'
+ elif conf.env.DEST_OS == 'darwin':
+ module_prefix = 'lib'
+ module_ext += '.dylib'
+ else:
+ module_prefix = 'lib'
+ module_ext += '.so'
+
+ autowaf.define(conf, 'SUIL_MODULE_PREFIX', module_prefix)
+ autowaf.define(conf, 'SUIL_MODULE_EXT', module_ext)
+
+ autowaf.set_lib_env(conf, 'suil', SUIL_VERSION)
+ conf.write_config_header('suil_config.h', remove=False)
+
+ autowaf.display_summary(conf)
+ autowaf.display_msg(conf, 'Static library', bool(conf.env.BUILD_STATIC))
+ autowaf.display_msg(conf, 'Shared library', bool(conf.env.BUILD_SHARED))
+
+ if conf.env.HAVE_GTK2:
+ autowaf.display_msg(conf, "Gtk2 Library Name",
+ conf.env.SUIL_GTK2_LIB_NAME)
+ if conf.env.HAVE_GTK3:
+ autowaf.display_msg(conf, "Gtk3 Library Name",
+ conf.env.SUIL_GTK3_LIB_NAME)
+
+ # Print summary message for every potentially supported wrapper
+ wrappers = [('cocoa', 'gtk2'),
+ ('gtk2', 'qt4'),
+ ('gtk2', 'qt5'),
+ ('qt4', 'gtk2'),
+ ('qt5', 'gtk2'),
+ ('win', 'gtk2'),
+ ('x11', 'gtk2'),
+ ('x11', 'gtk3'),
+ ('x11', 'qt4'),
+ ('x11', 'qt5'),
+ ('cocoa', 'qt5')]
+ for w in wrappers:
+ var = 'SUIL_WITH_%s_IN_%s' % (w[0].upper(), w[1].upper())
+ autowaf.display_msg(conf, 'Support for %s in %s' % (w[0], w[1]),
+ bool(conf.env[var]))
+
+ print('')
+
+def build(bld):
+ # C Headers
+ includedir = '${INCLUDEDIR}/suil-%s/suil' % SUIL_MAJOR_VERSION
+ bld.install_files(includedir, bld.path.ant_glob('suil/*.h'))
+ TaskGen.task_gen.mappings['.mm'] = TaskGen.task_gen.mappings['.cc']
+
+ # Pkgconfig file
+ autowaf.build_pc(bld, 'SUIL', SUIL_VERSION, SUIL_MAJOR_VERSION, [],
+ {'SUIL_MAJOR_VERSION' : SUIL_MAJOR_VERSION})
+
+ cflags = []
+ lib = []
+ modlib = []
+ if bld.env.DEST_OS == 'win32':
+ modlib += ['user32']
+ else:
+ cflags += ['-fvisibility=hidden']
+ if bld.is_defined('HAVE_LIBDL'):
+ lib += ['dl']
+ modlib += ['dl']
+
+ module_dir = '${LIBDIR}/suil-' + SUIL_MAJOR_VERSION
+
+ # Shared Library
+ if bld.env.BUILD_SHARED:
+ obj = bld(features = 'c cshlib',
+ export_includes = ['.'],
+ source = 'src/host.c src/instance.c',
+ target = 'suil-%s' % SUIL_MAJOR_VERSION,
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ name = 'libsuil',
+ vnum = SUIL_VERSION,
+ install_path = '${LIBDIR}',
+ cflags = cflags,
+ lib = lib,
+ uselib = 'LV2')
+
+ # Static library
+ if bld.env.BUILD_STATIC:
+ obj = bld(features = 'c cstlib',
+ export_includes = ['.'],
+ source = 'src/host.c src/instance.c',
+ target = 'suil-%s' % SUIL_MAJOR_VERSION,
+ includes = ['.'],
+ defines = ['SUIL_INTERNAL'],
+ name = 'libsuil_static',
+ vnum = SUIL_VERSION,
+ install_path = '${LIBDIR}',
+ cflags = cflags,
+ lib = lib,
+ uselib = 'LV2')
+
+ if bld.env.SUIL_WITH_GTK2_IN_QT4:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/gtk2_in_qt4.cpp',
+ target = 'suil_gtk2_in_qt4',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cxxflags = cflags,
+ lib = modlib)
+ autowaf.use_lib(bld, obj, 'GTK2 QT4 LV2')
+
+ if bld.env.SUIL_WITH_GTK2_IN_QT5:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/gtk2_in_qt5.cpp',
+ target = 'suil_gtk2_in_qt5',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cxxflags = cflags,
+ lib = modlib)
+ autowaf.use_lib(bld, obj, 'GTK2 QT5 LV2')
+
+ if bld.env.SUIL_WITH_QT4_IN_GTK2:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/qt4_in_gtk2.cpp',
+ target = 'suil_qt4_in_gtk2',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cxxflags = cflags,
+ lib = modlib,
+ linkflags = bld.env.NODELETE_FLAGS)
+ autowaf.use_lib(bld, obj, 'GTK2 QT4 LV2')
+
+ if bld.env.SUIL_WITH_QT5_IN_GTK2:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/qt5_in_gtk2.cpp',
+ target = 'suil_qt5_in_gtk2',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cxxflags = cflags,
+ lib = modlib,
+ linkflags = bld.env.NODELETE_FLAGS)
+ autowaf.use_lib(bld, obj, 'GTK2 QT5 LV2')
+
+ if bld.env.SUIL_WITH_X11_IN_GTK2:
+ obj = bld(features = 'c cshlib',
+ source = 'src/x11_in_gtk2.c',
+ target = 'suil_x11_in_gtk2',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib + ['X11'],
+ linkflags = bld.env.NODELETE_FLAGS)
+ autowaf.use_lib(bld, obj, 'GTK2 GTK2_X11 LV2')
+
+ if bld.env.SUIL_WITH_X11_IN_GTK3:
+ obj = bld(features = 'c cshlib',
+ source = 'src/x11_in_gtk3.c',
+ target = 'suil_x11_in_gtk3',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib + ['X11'],
+ linkflags = bld.env.NODELETE_FLAGS)
+ autowaf.use_lib(bld, obj, 'GTK3 GTK3_X11 LV2')
+
+ if bld.env.SUIL_WITH_COCOA_IN_GTK2:
+ obj = bld(features = 'cxx cshlib',
+ source = 'src/cocoa_in_gtk2.mm',
+ target = 'suil_cocoa_in_gtk2',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib,
+ linkflags = ['-framework', 'Cocoa'])
+ autowaf.use_lib(bld, obj, 'GTK2 LV2')
+
+ if bld.env.SUIL_WITH_WIN_IN_GTK2:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/win_in_gtk2.cpp',
+ target = 'suil_win_in_gtk2',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib,
+ linkflags = bld.env.NODELETE_FLAGS)
+ autowaf.use_lib(bld, obj, 'GTK2 LV2')
+
+ if bld.env.SUIL_WITH_X11_IN_QT4:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/x11_in_qt4.cpp',
+ target = 'suil_x11_in_qt4',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib)
+ autowaf.use_lib(bld, obj, 'QT4 LV2')
+
+ if bld.env.SUIL_WITH_X11_IN_QT5:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/x11_in_qt5.cpp',
+ target = 'suil_x11_in_qt5',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib)
+ autowaf.use_lib(bld, obj, 'QT5 LV2')
+
+ if bld.env.SUIL_WITH_COCOA_IN_QT5:
+ obj = bld(features = 'cxx cxxshlib',
+ source = 'src/cocoa_in_qt5.mm',
+ target = 'suil_cocoa_in_qt5',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib,
+ linkflags = ['-framework', 'Cocoa'])
+ autowaf.use_lib(bld, obj, 'QT5 QT5_MAC_EXTRAS LV2')
+
+ if bld.env.SUIL_WITH_X11:
+ obj = bld(features = 'c cshlib',
+ source = 'src/x11.c',
+ target = 'suil_x11',
+ includes = ['.'],
+ defines = ['SUIL_SHARED', 'SUIL_INTERNAL'],
+ install_path = module_dir,
+ cflags = cflags,
+ lib = modlib)
+ autowaf.use_lib(bld, obj, 'X11 LV2')
+
+ # Documentation
+ autowaf.build_dox(bld, 'SUIL', SUIL_VERSION, top, out)
+
+ bld.add_post_fun(autowaf.run_ldconfig)
+ if bld.env.DOCS:
+ bld.add_post_fun(lambda ctx: autowaf.make_simple_dox(APPNAME))
+
+def lint(ctx):
+ "checks code for style issues"
+ import subprocess
+ cmd = ("clang-tidy -p=. -header-filter=suil/ -checks=\"*," +
+ "-clang-analyzer-alpha.*," +
+ "-cppcoreguidelines-*," +
+ "-google-readability-todo," +
+ "-llvm-header-guard," +
+ "-llvm-include-order," +
+ "-misc-unused-parameters," +
+ "-misc-unused-parameters," +
+ "-modernize-*," +
+ "-readability-else-after-return," +
+ "-readability-implicit-bool-cast\" " +
+ "$(find .. -name '*.c' -or -name '*.cpp' -or -name '*.mm')")
+ subprocess.call(cmd, cwd='build', shell=True)
+
+def release(ctx):
+ autowaf.release(APPNAME.title(), VERSION)
+
+def upload(ctx):
+ autowaf.run_script(
+ ['scp suil-%s.tar* drobilla@drobilla.net:~/download.drobilla.net/' % VERSION,
+ 'rsync -ravz --delete -e ssh build/doc/html/ drobilla@drobilla.net:~/drobilla.net/docs/suil/'])
+
+def posts(ctx):
+ path = str(ctx.path.abspath())
+ autowaf.news_to_posts(
+ os.path.join(path, 'NEWS'),
+ {'title' : 'Suil',
+ 'description' : autowaf.get_blurb(os.path.join(path, 'README')),
+ 'dist_pattern' : 'http://download.drobilla.net/suil-%s.tar.bz2'},
+ { 'Author' : 'drobilla',
+ 'Tags' : 'Hacking, LAD, LV2, Suil' },
+ os.path.join(out, 'posts'))