360 lines
13 KiB
Python
360 lines
13 KiB
Python
|
import contextlib
|
||
|
import importlib
|
||
|
import os
|
||
|
import shutil
|
||
|
import subprocess
|
||
|
import sys
|
||
|
import tempfile
|
||
|
from unittest import skip
|
||
|
from ctypes import *
|
||
|
|
||
|
import numpy as np
|
||
|
|
||
|
import llvmlite.binding as ll
|
||
|
|
||
|
from numba.core import utils
|
||
|
from numba.tests.support import (TestCase, tag, import_dynamic, temp_directory,
|
||
|
has_blas, needs_setuptools)
|
||
|
|
||
|
import unittest
|
||
|
|
||
|
|
||
|
_skip_reason = 'windows only'
|
||
|
_windows_only = unittest.skipIf(not sys.platform.startswith('win'),
|
||
|
_skip_reason)
|
||
|
|
||
|
|
||
|
base_path = os.path.dirname(os.path.abspath(__file__))
|
||
|
|
||
|
|
||
|
def unset_macosx_deployment_target():
|
||
|
"""Unset MACOSX_DEPLOYMENT_TARGET because we are not building portable
|
||
|
libraries
|
||
|
"""
|
||
|
if 'MACOSX_DEPLOYMENT_TARGET' in os.environ:
|
||
|
del os.environ['MACOSX_DEPLOYMENT_TARGET']
|
||
|
|
||
|
|
||
|
@needs_setuptools
|
||
|
class TestCompilerChecks(TestCase):
|
||
|
|
||
|
# NOTE: THIS TEST MUST ALWAYS RUN ON WINDOWS, DO NOT SKIP
|
||
|
@_windows_only
|
||
|
def test_windows_compiler_validity(self):
|
||
|
# When inside conda-build VSINSTALLDIR should be set and windows should
|
||
|
# have a valid compiler available, `external_compiler_works()` should
|
||
|
# agree with this. If this is not the case then error out to alert devs.
|
||
|
|
||
|
# This is a local import to avoid deprecation warnings being generated
|
||
|
# through the use of the numba.pycc module.
|
||
|
from numba.pycc.platform import external_compiler_works
|
||
|
is_running_conda_build = os.environ.get('CONDA_BUILD', None) is not None
|
||
|
if is_running_conda_build:
|
||
|
if os.environ.get('VSINSTALLDIR', None) is not None:
|
||
|
self.assertTrue(external_compiler_works())
|
||
|
|
||
|
|
||
|
class BasePYCCTest(TestCase):
|
||
|
|
||
|
def setUp(self):
|
||
|
unset_macosx_deployment_target()
|
||
|
|
||
|
self.tmpdir = temp_directory('test_pycc')
|
||
|
# Make sure temporary files and directories created by
|
||
|
# distutils don't clutter the top-level /tmp
|
||
|
tempfile.tempdir = self.tmpdir
|
||
|
|
||
|
def tearDown(self):
|
||
|
tempfile.tempdir = None
|
||
|
# Since we're executing the module-under-test several times
|
||
|
# from the same process, we must clear the exports registry
|
||
|
# between invocations.
|
||
|
|
||
|
# This is a local import to avoid deprecation warnings being generated
|
||
|
# through the use of the numba.pycc module.
|
||
|
from numba.pycc.decorators import clear_export_registry
|
||
|
clear_export_registry()
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def check_c_ext(self, extdir, name):
|
||
|
sys.path.append(extdir)
|
||
|
try:
|
||
|
lib = import_dynamic(name)
|
||
|
yield lib
|
||
|
finally:
|
||
|
sys.path.remove(extdir)
|
||
|
sys.modules.pop(name, None)
|
||
|
|
||
|
|
||
|
@needs_setuptools
|
||
|
class TestCC(BasePYCCTest):
|
||
|
|
||
|
def setUp(self):
|
||
|
super(TestCC, self).setUp()
|
||
|
self.skip_if_no_external_compiler() # external compiler needed
|
||
|
from numba.tests import compile_with_pycc
|
||
|
self._test_module = compile_with_pycc
|
||
|
importlib.reload(self._test_module)
|
||
|
|
||
|
@contextlib.contextmanager
|
||
|
def check_cc_compiled(self, cc):
|
||
|
#cc.verbose = True
|
||
|
cc.output_dir = self.tmpdir
|
||
|
cc.compile()
|
||
|
|
||
|
with self.check_c_ext(self.tmpdir, cc.name) as lib:
|
||
|
yield lib
|
||
|
|
||
|
def check_cc_compiled_in_subprocess(self, lib, code):
|
||
|
prolog = """if 1:
|
||
|
import sys
|
||
|
import types
|
||
|
# to disable numba package
|
||
|
sys.modules['numba'] = types.ModuleType('numba')
|
||
|
try:
|
||
|
from numba import njit
|
||
|
except ImportError:
|
||
|
pass
|
||
|
else:
|
||
|
raise RuntimeError('cannot disable numba package')
|
||
|
|
||
|
sys.path.insert(0, %(path)r)
|
||
|
import %(name)s as lib
|
||
|
""" % {'name': lib.__name__,
|
||
|
'path': os.path.dirname(lib.__file__)}
|
||
|
code = prolog.strip(' ') + code
|
||
|
subprocess.check_call([sys.executable, '-c', code])
|
||
|
|
||
|
def test_cc_properties(self):
|
||
|
cc = self._test_module.cc
|
||
|
self.assertEqual(cc.name, 'pycc_test_simple')
|
||
|
|
||
|
# Inferred output directory
|
||
|
d = self._test_module.cc.output_dir
|
||
|
self.assertTrue(os.path.isdir(d), d)
|
||
|
|
||
|
# Inferred output filename
|
||
|
f = self._test_module.cc.output_file
|
||
|
self.assertFalse(os.path.exists(f), f)
|
||
|
self.assertTrue(os.path.basename(f).startswith('pycc_test_simple.'), f)
|
||
|
if sys.platform.startswith('linux'):
|
||
|
self.assertTrue(f.endswith('.so'), f)
|
||
|
# This is a local import to avoid deprecation warnings being
|
||
|
# generated through the use of the numba.pycc module.
|
||
|
from numba.pycc.platform import find_pyext_ending
|
||
|
self.assertIn(find_pyext_ending(), f)
|
||
|
|
||
|
def test_compile(self):
|
||
|
with self.check_cc_compiled(self._test_module.cc) as lib:
|
||
|
res = lib.multi(123, 321)
|
||
|
self.assertPreciseEqual(res, 123 * 321)
|
||
|
res = lib.multf(987, 321)
|
||
|
self.assertPreciseEqual(res, 987.0 * 321.0)
|
||
|
res = lib.square(5)
|
||
|
self.assertPreciseEqual(res, 25)
|
||
|
self.assertIs(lib.get_none(), None)
|
||
|
with self.assertRaises(ZeroDivisionError):
|
||
|
lib.div(1, 0)
|
||
|
|
||
|
def check_compile_for_cpu(self, cpu_name):
|
||
|
cc = self._test_module.cc
|
||
|
cc.target_cpu = cpu_name
|
||
|
|
||
|
with self.check_cc_compiled(cc) as lib:
|
||
|
res = lib.multi(123, 321)
|
||
|
self.assertPreciseEqual(res, 123 * 321)
|
||
|
self.assertEqual(lib.multi.__module__, 'pycc_test_simple')
|
||
|
|
||
|
def test_compile_for_cpu(self):
|
||
|
# Compiling for the host CPU should always succeed
|
||
|
self.check_compile_for_cpu(ll.get_host_cpu_name())
|
||
|
|
||
|
def test_compile_for_cpu_host(self):
|
||
|
# Compiling for the host CPU should always succeed
|
||
|
self.check_compile_for_cpu("host")
|
||
|
|
||
|
def test_compile_helperlib(self):
|
||
|
with self.check_cc_compiled(self._test_module.cc_helperlib) as lib:
|
||
|
res = lib.power(2, 7)
|
||
|
self.assertPreciseEqual(res, 128)
|
||
|
for val in (-1, -1 + 0j, np.complex128(-1)):
|
||
|
res = lib.sqrt(val)
|
||
|
self.assertPreciseEqual(res, 1j)
|
||
|
for val in (4, 4.0, np.float64(4)):
|
||
|
res = lib.np_sqrt(val)
|
||
|
self.assertPreciseEqual(res, 2.0)
|
||
|
res = lib.spacing(1.0)
|
||
|
self.assertPreciseEqual(res, 2**-52)
|
||
|
# Implicit seeding at startup should guarantee a non-pathological
|
||
|
# start state.
|
||
|
self.assertNotEqual(lib.random(-1), lib.random(-1))
|
||
|
res = lib.random(42)
|
||
|
expected = np.random.RandomState(42).random_sample()
|
||
|
self.assertPreciseEqual(res, expected)
|
||
|
res = lib.size(np.float64([0] * 3))
|
||
|
self.assertPreciseEqual(res, 3)
|
||
|
|
||
|
code = """if 1:
|
||
|
from numpy.testing import assert_equal, assert_allclose
|
||
|
res = lib.power(2, 7)
|
||
|
assert res == 128
|
||
|
res = lib.random(42)
|
||
|
assert_allclose(res, %(expected)s)
|
||
|
res = lib.spacing(1.0)
|
||
|
assert_allclose(res, 2**-52)
|
||
|
""" % {'expected': expected}
|
||
|
self.check_cc_compiled_in_subprocess(lib, code)
|
||
|
|
||
|
def test_compile_nrt(self):
|
||
|
with self.check_cc_compiled(self._test_module.cc_nrt) as lib:
|
||
|
# Sanity check
|
||
|
self.assertPreciseEqual(lib.zero_scalar(1), 0.0)
|
||
|
res = lib.zeros(3)
|
||
|
self.assertEqual(list(res), [0, 0, 0])
|
||
|
if has_blas:
|
||
|
res = lib.vector_dot(4)
|
||
|
self.assertPreciseEqual(res, 30.0)
|
||
|
# test argsort
|
||
|
val = np.float64([2., 5., 1., 3., 4.])
|
||
|
res = lib.np_argsort(val)
|
||
|
expected = np.argsort(val)
|
||
|
self.assertPreciseEqual(res, expected)
|
||
|
|
||
|
code = """if 1:
|
||
|
from numpy.testing import assert_equal
|
||
|
from numpy import float64, argsort
|
||
|
res = lib.zero_scalar(1)
|
||
|
assert res == 0.0
|
||
|
res = lib.zeros(3)
|
||
|
assert list(res) == [0, 0, 0]
|
||
|
if %(has_blas)s:
|
||
|
res = lib.vector_dot(4)
|
||
|
assert res == 30.0
|
||
|
val = float64([2., 5., 1., 3., 4.])
|
||
|
res = lib.np_argsort(val)
|
||
|
expected = argsort(val)
|
||
|
assert_equal(res, expected)
|
||
|
""" % dict(has_blas=has_blas)
|
||
|
self.check_cc_compiled_in_subprocess(lib, code)
|
||
|
|
||
|
def test_hashing(self):
|
||
|
with self.check_cc_compiled(self._test_module.cc_nrt) as lib:
|
||
|
res = lib.hash_literal_str_A()
|
||
|
self.assertPreciseEqual(res, hash("A"))
|
||
|
res = lib.hash_str("A")
|
||
|
self.assertPreciseEqual(res, hash("A"))
|
||
|
|
||
|
code = """if 1:
|
||
|
from numpy.testing import assert_equal
|
||
|
res = lib.hash_literal_str_A()
|
||
|
assert_equal(res, hash("A"))
|
||
|
res = lib.hash_str("A")
|
||
|
assert_equal(res, hash("A"))
|
||
|
"""
|
||
|
self.check_cc_compiled_in_subprocess(lib, code)
|
||
|
|
||
|
def test_c_extension_usecase(self):
|
||
|
# Test C-extensions
|
||
|
with self.check_cc_compiled(self._test_module.cc_nrt) as lib:
|
||
|
arr = np.arange(128, dtype=np.intp)
|
||
|
got = lib.dict_usecase(arr)
|
||
|
expect = arr * arr
|
||
|
self.assertPreciseEqual(got, expect)
|
||
|
|
||
|
|
||
|
@needs_setuptools
|
||
|
class TestDistutilsSupport(TestCase):
|
||
|
|
||
|
def setUp(self):
|
||
|
super().setUp()
|
||
|
self.skip_if_no_external_compiler() # external compiler needed
|
||
|
|
||
|
unset_macosx_deployment_target()
|
||
|
|
||
|
# Copy the test project into a temp directory to avoid
|
||
|
# keeping any build leftovers in the source tree
|
||
|
self.tmpdir = temp_directory('test_pycc_distutils')
|
||
|
source_dir = os.path.join(base_path, 'pycc_distutils_usecase')
|
||
|
self.usecase_dir = os.path.join(self.tmpdir, 'work')
|
||
|
shutil.copytree(source_dir, self.usecase_dir)
|
||
|
|
||
|
def check_setup_py(self, setup_py_file):
|
||
|
# Compute PYTHONPATH to ensure the child processes see this Numba
|
||
|
import numba
|
||
|
numba_path = os.path.abspath(os.path.dirname(
|
||
|
os.path.dirname(numba.__file__)))
|
||
|
env = dict(os.environ)
|
||
|
if env.get('PYTHONPATH', ''):
|
||
|
env['PYTHONPATH'] = numba_path + os.pathsep + env['PYTHONPATH']
|
||
|
else:
|
||
|
env['PYTHONPATH'] = numba_path
|
||
|
|
||
|
def run_python(args):
|
||
|
p = subprocess.Popen([sys.executable] + args,
|
||
|
cwd=self.usecase_dir,
|
||
|
stdout=subprocess.PIPE,
|
||
|
stderr=subprocess.STDOUT,
|
||
|
env=env)
|
||
|
out, _ = p.communicate()
|
||
|
rc = p.wait()
|
||
|
if rc != 0:
|
||
|
self.fail("python failed with the following output:\n%s"
|
||
|
% out.decode('utf-8', 'ignore'))
|
||
|
|
||
|
run_python([setup_py_file, "build_ext", "--inplace"])
|
||
|
code = """if 1:
|
||
|
import pycc_compiled_module as lib
|
||
|
assert lib.get_const() == 42
|
||
|
res = lib.ones(3)
|
||
|
assert list(res) == [1.0, 1.0, 1.0]
|
||
|
"""
|
||
|
run_python(["-c", code])
|
||
|
|
||
|
def check_setup_nested_py(self, setup_py_file):
|
||
|
# Compute PYTHONPATH to ensure the child processes see this Numba
|
||
|
import numba
|
||
|
numba_path = os.path.abspath(os.path.dirname(
|
||
|
os.path.dirname(numba.__file__)))
|
||
|
env = dict(os.environ)
|
||
|
if env.get('PYTHONPATH', ''):
|
||
|
env['PYTHONPATH'] = numba_path + os.pathsep + env['PYTHONPATH']
|
||
|
else:
|
||
|
env['PYTHONPATH'] = numba_path
|
||
|
|
||
|
def run_python(args):
|
||
|
p = subprocess.Popen([sys.executable] + args,
|
||
|
cwd=self.usecase_dir,
|
||
|
stdout=subprocess.PIPE,
|
||
|
stderr=subprocess.STDOUT,
|
||
|
env=env)
|
||
|
out, _ = p.communicate()
|
||
|
rc = p.wait()
|
||
|
if rc != 0:
|
||
|
self.fail("python failed with the following output:\n%s"
|
||
|
% out.decode('utf-8', 'ignore'))
|
||
|
|
||
|
run_python([setup_py_file, "build_ext", "--inplace"])
|
||
|
code = """if 1:
|
||
|
import nested.pycc_compiled_module as lib
|
||
|
assert lib.get_const() == 42
|
||
|
res = lib.ones(3)
|
||
|
assert list(res) == [1.0, 1.0, 1.0]
|
||
|
"""
|
||
|
run_python(["-c", code])
|
||
|
|
||
|
def test_setup_py_distutils(self):
|
||
|
self.check_setup_py("setup_distutils.py")
|
||
|
|
||
|
def test_setup_py_distutils_nested(self):
|
||
|
self.check_setup_nested_py("setup_distutils_nested.py")
|
||
|
|
||
|
def test_setup_py_setuptools(self):
|
||
|
self.check_setup_py("setup_setuptools.py")
|
||
|
|
||
|
def test_setup_py_setuptools_nested(self):
|
||
|
self.check_setup_nested_py("setup_setuptools_nested.py")
|
||
|
|
||
|
|
||
|
if __name__ == "__main__":
|
||
|
unittest.main()
|