import itertools import numpy as np import re from numba import cuda, int64 from numba.cuda import compile_ptx from numba.core.types import f2 from numba.cuda.testing import (unittest, CUDATestCase, skip_on_cudasim, skip_unless_cc_53) def simple_threadidx(ary): i = cuda.threadIdx.x ary[0] = i def fill_threadidx(ary): i = cuda.threadIdx.x ary[i] = i def fill3d_threadidx(ary): i = cuda.threadIdx.x j = cuda.threadIdx.y k = cuda.threadIdx.z ary[i, j, k] = (i + 1) * (j + 1) * (k + 1) def simple_grid1d(ary): i = cuda.grid(1) ary[i] = i def simple_grid2d(ary): i, j = cuda.grid(2) ary[i, j] = i + j def simple_gridsize1d(ary): i = cuda.grid(1) x = cuda.gridsize(1) if i == 0: ary[0] = x def simple_gridsize2d(ary): i, j = cuda.grid(2) x, y = cuda.gridsize(2) if i == 0 and j == 0: ary[0] = x ary[1] = y def intrinsic_forloop_step(c): startX, startY = cuda.grid(2) gridX = cuda.gridDim.x * cuda.blockDim.x gridY = cuda.gridDim.y * cuda.blockDim.y height, width = c.shape for x in range(startX, width, gridX): for y in range(startY, height, gridY): c[y, x] = x + y def simple_popc(ary, c): ary[0] = cuda.popc(c) def simple_fma(ary, a, b, c): ary[0] = cuda.fma(a, b, c) def simple_hadd(ary, a, b): ary[0] = cuda.fp16.hadd(a[0], b[0]) def simple_hadd_scalar(ary, a, b): ary[0] = cuda.fp16.hadd(a, b) def simple_hfma(ary, a, b, c): ary[0] = cuda.fp16.hfma(a[0], b[0], c[0]) def simple_hfma_scalar(ary, a, b, c): ary[0] = cuda.fp16.hfma(a, b, c) def simple_hsub(ary, a, b): ary[0] = cuda.fp16.hsub(a[0], b[0]) def simple_hsub_scalar(ary, a, b): ary[0] = cuda.fp16.hsub(a, b) def simple_hmul(ary, a, b): ary[0] = cuda.fp16.hmul(a[0], b[0]) def simple_hmul_scalar(ary, a, b): ary[0] = cuda.fp16.hmul(a, b) def simple_hneg(ary, a): ary[0] = cuda.fp16.hneg(a[0]) def simple_hneg_scalar(ary, a): ary[0] = cuda.fp16.hneg(a) def simple_habs(ary, a): ary[0] = cuda.fp16.habs(a[0]) def simple_habs_scalar(ary, a): ary[0] = cuda.fp16.habs(a) def simple_cbrt(ary, a): ary[0] = cuda.cbrt(a) def simple_brev(ary, c): ary[0] = cuda.brev(c) def simple_clz(ary, c): ary[0] = cuda.clz(c) def simple_ffs(ary, c): ary[0] = cuda.ffs(c) def simple_round(ary, c): ary[0] = round(c) def simple_round_to(ary, c, ndigits): ary[0] = round(c, ndigits) def branching_with_ifs(a, b, c): i = cuda.grid(1) if a[i] > 4: if b % 2 == 0: a[i] = c[i] else: a[i] = 13 else: a[i] = 3 def branching_with_selps(a, b, c): i = cuda.grid(1) inner = cuda.selp(b % 2 == 0, c[i], 13) a[i] = cuda.selp(a[i] > 4, inner, 3) def simple_laneid(ary): i = cuda.grid(1) ary[i] = cuda.laneid def simple_warpsize(ary): ary[0] = cuda.warpsize class TestCudaIntrinsic(CUDATestCase): def test_simple_threadidx(self): compiled = cuda.jit("void(int32[:])")(simple_threadidx) ary = np.ones(1, dtype=np.int32) compiled[1, 1](ary) self.assertTrue(ary[0] == 0) def test_fill_threadidx(self): compiled = cuda.jit("void(int32[:])")(fill_threadidx) N = 10 ary = np.ones(N, dtype=np.int32) exp = np.arange(N, dtype=np.int32) compiled[1, N](ary) self.assertTrue(np.all(ary == exp)) def test_fill3d_threadidx(self): X, Y, Z = 4, 5, 6 def c_contigous(): compiled = cuda.jit("void(int32[:,:,::1])")(fill3d_threadidx) ary = np.zeros((X, Y, Z), dtype=np.int32) compiled[1, (X, Y, Z)](ary) return ary def f_contigous(): compiled = cuda.jit("void(int32[::1,:,:])")(fill3d_threadidx) ary = np.asfortranarray(np.zeros((X, Y, Z), dtype=np.int32)) compiled[1, (X, Y, Z)](ary) return ary c_res = c_contigous() f_res = f_contigous() self.assertTrue(np.all(c_res == f_res)) def test_simple_grid1d(self): compiled = cuda.jit("void(int32[::1])")(simple_grid1d) ntid, nctaid = 3, 7 nelem = ntid * nctaid ary = np.empty(nelem, dtype=np.int32) compiled[nctaid, ntid](ary) self.assertTrue(np.all(ary == np.arange(nelem))) def test_simple_grid2d(self): compiled = cuda.jit("void(int32[:,::1])")(simple_grid2d) ntid = (4, 3) nctaid = (5, 6) shape = (ntid[0] * nctaid[0], ntid[1] * nctaid[1]) ary = np.empty(shape, dtype=np.int32) exp = ary.copy() compiled[nctaid, ntid](ary) for i in range(ary.shape[0]): for j in range(ary.shape[1]): exp[i, j] = i + j self.assertTrue(np.all(ary == exp)) def test_simple_gridsize1d(self): compiled = cuda.jit("void(int32[::1])")(simple_gridsize1d) ntid, nctaid = 3, 7 ary = np.zeros(1, dtype=np.int32) compiled[nctaid, ntid](ary) self.assertEqual(ary[0], nctaid * ntid) @skip_on_cudasim('Tests PTX emission') def test_selp(self): sig = (int64[:], int64, int64[:]) cu_branching_with_ifs = cuda.jit(sig)(branching_with_ifs) cu_branching_with_selps = cuda.jit(sig)(branching_with_selps) n = 32 b = 6 c = np.full(shape=32, fill_value=17, dtype=np.int64) expected = c.copy() expected[:5] = 3 a = np.arange(n, dtype=np.int64) cu_branching_with_ifs[n, 1](a, b, c) ptx = cu_branching_with_ifs.inspect_asm(sig) self.assertEqual(2, len(re.findall(r'\s+bra\s+', ptx))) np.testing.assert_array_equal(a, expected, err_msg='branching') a = np.arange(n, dtype=np.int64) cu_branching_with_selps[n, 1](a, b, c) ptx = cu_branching_with_selps.inspect_asm(sig) self.assertEqual(0, len(re.findall(r'\s+bra\s+', ptx))) np.testing.assert_array_equal(a, expected, err_msg='selp') def test_simple_gridsize2d(self): compiled = cuda.jit("void(int32[::1])")(simple_gridsize2d) ntid = (4, 3) nctaid = (5, 6) ary = np.zeros(2, dtype=np.int32) compiled[nctaid, ntid](ary) self.assertEqual(ary[0], nctaid[0] * ntid[0]) self.assertEqual(ary[1], nctaid[1] * ntid[1]) def test_intrinsic_forloop_step(self): compiled = cuda.jit("void(int32[:,::1])")(intrinsic_forloop_step) ntid = (4, 3) nctaid = (5, 6) shape = (ntid[0] * nctaid[0], ntid[1] * nctaid[1]) ary = np.empty(shape, dtype=np.int32) compiled[nctaid, ntid](ary) gridX, gridY = shape height, width = ary.shape for i, j in zip(range(ntid[0]), range(ntid[1])): startX, startY = gridX + i, gridY + j for x in range(startX, width, gridX): for y in range(startY, height, gridY): self.assertTrue(ary[y, x] == x + y, (ary[y, x], x + y)) def test_3dgrid(self): @cuda.jit def foo(out): x, y, z = cuda.grid(3) a, b, c = cuda.gridsize(3) out[x, y, z] = a * b * c arr = np.zeros(9 ** 3, dtype=np.int32).reshape(9, 9, 9) foo[(3, 3, 3), (3, 3, 3)](arr) np.testing.assert_equal(arr, 9 ** 3) def test_3dgrid_2(self): @cuda.jit def foo(out): x, y, z = cuda.grid(3) a, b, c = cuda.gridsize(3) grid_is_right = ( x == cuda.threadIdx.x + cuda.blockIdx.x * cuda.blockDim.x and y == cuda.threadIdx.y + cuda.blockIdx.y * cuda.blockDim.y and z == cuda.threadIdx.z + cuda.blockIdx.z * cuda.blockDim.z ) gridsize_is_right = (a == cuda.blockDim.x * cuda.gridDim.x and b == cuda.blockDim.y * cuda.gridDim.y and c == cuda.blockDim.z * cuda.gridDim.z) out[x, y, z] = grid_is_right and gridsize_is_right x, y, z = (4 * 3, 3 * 2, 2 * 4) arr = np.zeros((x * y * z), dtype=np.bool_).reshape(x, y, z) foo[(4, 3, 2), (3, 2, 4)](arr) self.assertTrue(np.all(arr)) def test_popc_u4(self): compiled = cuda.jit("void(int32[:], uint32)")(simple_popc) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0xF0) self.assertEquals(ary[0], 4) def test_popc_u8(self): compiled = cuda.jit("void(int32[:], uint64)")(simple_popc) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0xF00000000000) self.assertEquals(ary[0], 4) def test_fma_f4(self): compiled = cuda.jit("void(f4[:], f4, f4, f4)")(simple_fma) ary = np.zeros(1, dtype=np.float32) compiled[1, 1](ary, 2., 3., 4.) np.testing.assert_allclose(ary[0], 2 * 3 + 4) def test_fma_f8(self): compiled = cuda.jit("void(f8[:], f8, f8, f8)")(simple_fma) ary = np.zeros(1, dtype=np.float64) compiled[1, 1](ary, 2., 3., 4.) np.testing.assert_allclose(ary[0], 2 * 3 + 4) @skip_unless_cc_53 def test_hadd(self): compiled = cuda.jit("void(f2[:], f2[:], f2[:])")(simple_hadd) ary = np.zeros(1, dtype=np.float16) arg1 = np.array([3.], dtype=np.float16) arg2 = np.array([4.], dtype=np.float16) compiled[1, 1](ary, arg1, arg2) np.testing.assert_allclose(ary[0], arg1 + arg2) @skip_unless_cc_53 def test_hadd_scalar(self): compiled = cuda.jit("void(f2[:], f2, f2)")(simple_hadd_scalar) ary = np.zeros(1, dtype=np.float16) arg1 = np.float16(3.1415926) arg2 = np.float16(3.) compiled[1, 1](ary, arg1, arg2) ref = arg1 + arg2 np.testing.assert_allclose(ary[0], ref) @skip_on_cudasim('Compilation unsupported in the simulator') def test_hadd_ptx(self): args = (f2[:], f2, f2) ptx, _ = compile_ptx(simple_hadd_scalar, args, cc=(5, 3)) self.assertIn('add.f16', ptx) @skip_unless_cc_53 def test_hfma(self): compiled = cuda.jit("void(f2[:], f2[:], f2[:], f2[:])")(simple_hfma) ary = np.zeros(1, dtype=np.float16) arg1 = np.array([2.], dtype=np.float16) arg2 = np.array([3.], dtype=np.float16) arg3 = np.array([4.], dtype=np.float16) compiled[1, 1](ary, arg1, arg2, arg3) np.testing.assert_allclose(ary[0], arg1 * arg2 + arg3) @skip_unless_cc_53 def test_hfma_scalar(self): compiled = cuda.jit("void(f2[:], f2, f2, f2)")(simple_hfma_scalar) ary = np.zeros(1, dtype=np.float16) arg1 = np.float16(2.) arg2 = np.float16(3.) arg3 = np.float16(4.) compiled[1, 1](ary, arg1, arg2, arg3) ref = arg1 * arg2 + arg3 np.testing.assert_allclose(ary[0], ref) @skip_on_cudasim('Compilation unsupported in the simulator') def test_hfma_ptx(self): args = (f2[:], f2, f2, f2) ptx, _ = compile_ptx(simple_hfma_scalar, args, cc=(5, 3)) self.assertIn('fma.rn.f16', ptx) @skip_unless_cc_53 def test_hsub(self): compiled = cuda.jit("void(f2[:], f2[:], f2[:])")(simple_hsub) ary = np.zeros(1, dtype=np.float16) arg1 = np.array([3.], dtype=np.float16) arg2 = np.array([4.], dtype=np.float16) compiled[1, 1](ary, arg1, arg2) np.testing.assert_allclose(ary[0], arg1 - arg2) @skip_unless_cc_53 def test_hsub_scalar(self): compiled = cuda.jit("void(f2[:], f2, f2)")(simple_hsub_scalar) ary = np.zeros(1, dtype=np.float16) arg1 = np.float16(3.1415926) arg2 = np.float16(1.57) compiled[1, 1](ary, arg1, arg2) ref = arg1 - arg2 np.testing.assert_allclose(ary[0], ref) @skip_on_cudasim('Compilation unsupported in the simulator') def test_hsub_ptx(self): args = (f2[:], f2, f2) ptx, _ = compile_ptx(simple_hsub_scalar, args, cc=(5, 3)) self.assertIn('sub.f16', ptx) @skip_unless_cc_53 def test_hmul(self): compiled = cuda.jit()(simple_hmul) ary = np.zeros(1, dtype=np.float16) arg1 = np.array([3.], dtype=np.float16) arg2 = np.array([4.], dtype=np.float16) compiled[1, 1](ary, arg1, arg2) np.testing.assert_allclose(ary[0], arg1 * arg2) @skip_unless_cc_53 def test_hmul_scalar(self): compiled = cuda.jit("void(f2[:], f2, f2)")(simple_hmul_scalar) ary = np.zeros(1, dtype=np.float16) arg1 = np.float16(3.1415926) arg2 = np.float16(1.57) compiled[1, 1](ary, arg1, arg2) ref = arg1 * arg2 np.testing.assert_allclose(ary[0], ref) @skip_on_cudasim('Compilation unsupported in the simulator') def test_hmul_ptx(self): args = (f2[:], f2, f2) ptx, _ = compile_ptx(simple_hmul_scalar, args, cc=(5, 3)) self.assertIn('mul.f16', ptx) @skip_unless_cc_53 def test_hneg(self): compiled = cuda.jit("void(f2[:], f2[:])")(simple_hneg) ary = np.zeros(1, dtype=np.float16) arg1 = np.array([3.], dtype=np.float16) compiled[1, 1](ary, arg1) np.testing.assert_allclose(ary[0], -arg1) @skip_unless_cc_53 def test_hneg_scalar(self): compiled = cuda.jit("void(f2[:], f2)")(simple_hneg_scalar) ary = np.zeros(1, dtype=np.float16) arg1 = np.float16(3.1415926) compiled[1, 1](ary, arg1) ref = -arg1 np.testing.assert_allclose(ary[0], ref) @skip_on_cudasim('Compilation unsupported in the simulator') def test_hneg_ptx(self): args = (f2[:], f2) ptx, _ = compile_ptx(simple_hneg_scalar, args, cc=(5, 3)) self.assertIn('neg.f16', ptx) @skip_unless_cc_53 def test_habs(self): compiled = cuda.jit()(simple_habs) ary = np.zeros(1, dtype=np.float16) arg1 = np.array([-3.], dtype=np.float16) compiled[1, 1](ary, arg1) np.testing.assert_allclose(ary[0], abs(arg1)) @skip_unless_cc_53 def test_habs_scalar(self): compiled = cuda.jit("void(f2[:], f2)")(simple_habs_scalar) ary = np.zeros(1, dtype=np.float16) arg1 = np.float16(-3.1415926) compiled[1, 1](ary, arg1) ref = abs(arg1) np.testing.assert_allclose(ary[0], ref) @skip_on_cudasim('Compilation unsupported in the simulator') def test_habs_ptx(self): args = (f2[:], f2) ptx, _ = compile_ptx(simple_habs_scalar, args, cc=(5, 3)) if cuda.runtime.get_version() < (10, 2): self.assertRegex(ptx, r'and\.b16.*0x7FFF;') else: self.assertIn('abs.f16', ptx) def test_cbrt_f32(self): compiled = cuda.jit("void(float32[:], float32)")(simple_cbrt) ary = np.zeros(1, dtype=np.float32) cbrt_arg = 2. compiled[1, 1](ary, cbrt_arg) np.testing.assert_allclose(ary[0], cbrt_arg ** (1 / 3)) def test_cbrt_f64(self): compiled = cuda.jit("void(float64[:], float64)")(simple_cbrt) ary = np.zeros(1, dtype=np.float64) cbrt_arg = 6. compiled[1, 1](ary, cbrt_arg) np.testing.assert_allclose(ary[0], cbrt_arg ** (1 / 3)) def test_brev_u4(self): compiled = cuda.jit("void(uint32[:], uint32)")(simple_brev) ary = np.zeros(1, dtype=np.uint32) compiled[1, 1](ary, 0x000030F0) self.assertEquals(ary[0], 0x0F0C0000) @skip_on_cudasim('only get given a Python "int", assumes 32 bits') def test_brev_u8(self): compiled = cuda.jit("void(uint64[:], uint64)")(simple_brev) ary = np.zeros(1, dtype=np.uint64) compiled[1, 1](ary, 0x000030F0000030F0) self.assertEquals(ary[0], 0x0F0C00000F0C0000) def test_clz_i4(self): compiled = cuda.jit("void(int32[:], int32)")(simple_clz) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x00100000) self.assertEquals(ary[0], 11) def test_clz_u4(self): """ Although the CUDA Math API (http://docs.nvidia.com/cuda/cuda-math-api/group__CUDA__MATH__INTRINSIC__INT.html) only says int32 & int64 arguments are supported in C code, the LLVM IR input supports i8, i16, i32 & i64 (LLVM doesn't have a concept of unsigned integers, just unsigned operations on integers). http://docs.nvidia.com/cuda/nvvm-ir-spec/index.html#bit-manipulations-intrinics """ compiled = cuda.jit("void(int32[:], uint32)")(simple_clz) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x00100000) self.assertEquals(ary[0], 11) def test_clz_i4_1s(self): compiled = cuda.jit("void(int32[:], int32)")(simple_clz) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0xFFFFFFFF) self.assertEquals(ary[0], 0) def test_clz_i4_0s(self): compiled = cuda.jit("void(int32[:], int32)")(simple_clz) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x0) self.assertEquals(ary[0], 32, "CUDA semantics") @skip_on_cudasim('only get given a Python "int", assumes 32 bits') def test_clz_i8(self): compiled = cuda.jit("void(int32[:], int64)")(simple_clz) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x000000000010000) self.assertEquals(ary[0], 47) def test_ffs_i4(self): compiled = cuda.jit("void(int32[:], int32)")(simple_ffs) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x00100000) self.assertEquals(ary[0], 21) compiled[1, 1](ary, 0x80000000) self.assertEquals(ary[0], 32) def test_ffs_u4(self): compiled = cuda.jit("void(int32[:], uint32)")(simple_ffs) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x00100000) self.assertEquals(ary[0], 21) compiled[1, 1](ary, 0x80000000) self.assertEquals(ary[0], 32) def test_ffs_i4_1s(self): compiled = cuda.jit("void(int32[:], int32)")(simple_ffs) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0xFFFFFFFF) self.assertEquals(ary[0], 1) def test_ffs_i4_0s(self): compiled = cuda.jit("void(int32[:], int32)")(simple_ffs) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x0) self.assertEquals(ary[0], 0) @skip_on_cudasim('only get given a Python "int", assumes 32 bits') def test_ffs_i8(self): compiled = cuda.jit("void(int32[:], int64)")(simple_ffs) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary, 0x000000000010000) self.assertEquals(ary[0], 17) compiled[1, 1](ary, 0x100000000) self.assertEquals(ary[0], 33) def test_simple_laneid(self): compiled = cuda.jit("void(int32[:])")(simple_laneid) count = 2 ary = np.zeros(count * 32, dtype=np.int32) exp = np.tile(np.arange(32, dtype=np.int32), count) compiled[1, count * 32](ary) self.assertTrue(np.all(ary == exp)) def test_simple_warpsize(self): compiled = cuda.jit("void(int32[:])")(simple_warpsize) ary = np.zeros(1, dtype=np.int32) compiled[1, 1](ary) self.assertEquals(ary[0], 32, "CUDA semantics") def test_round_f4(self): compiled = cuda.jit("void(int64[:], float32)")(simple_round) ary = np.zeros(1, dtype=np.int64) for i in [-3.0, -2.5, -2.25, -1.5, 1.5, 2.25, 2.5, 2.75]: compiled[1, 1](ary, i) self.assertEquals(ary[0], round(i)) def test_round_f8(self): compiled = cuda.jit("void(int64[:], float64)")(simple_round) ary = np.zeros(1, dtype=np.int64) for i in [-3.0, -2.5, -2.25, -1.5, 1.5, 2.25, 2.5, 2.75]: compiled[1, 1](ary, i) self.assertEquals(ary[0], round(i)) def test_round_to_f4(self): compiled = cuda.jit("void(float32[:], float32, int32)")(simple_round_to) ary = np.zeros(1, dtype=np.float32) np.random.seed(123) vals = np.random.random(32).astype(np.float32) np.concatenate((vals, np.array([np.inf, -np.inf, np.nan]))) digits = ( # Common case branch of round_to_impl -5, -4, -3, -2, -1, 0, 1, 2, 3, 4, 5, # The algorithm currently implemented can only round to 13 digits # with single precision. Note that this doesn't trigger the # "overflow safe" branch of the implementation, which can only be # hit when using double precision. 13 ) for val, ndigits in itertools.product(vals, digits): with self.subTest(val=val, ndigits=ndigits): compiled[1, 1](ary, val, ndigits) self.assertPreciseEqual(ary[0], round(val, ndigits), prec='single') # CPython on most platforms uses rounding based on dtoa.c, whereas the CUDA # round-to implementation uses CPython's fallback implementation, which has # slightly different behavior at the edges of the domain. Since the CUDA # simulator executes using CPython, we need to skip this test when the # simulator is active. @skip_on_cudasim('Overflow behavior differs on CPython') def test_round_to_f4_overflow(self): # Test that the input value is returned when y in round_ndigits # overflows. compiled = cuda.jit("void(float32[:], float32, int32)")(simple_round_to) ary = np.zeros(1, dtype=np.float32) val = np.finfo(np.float32).max # An unusually large number of digits is required to hit the "y # overflows" branch of the implementation because the typing results in # the computation of y as float64. ndigits = 300 compiled[1, 1](ary, val, ndigits) self.assertEqual(ary[0], val) def test_round_to_f4_halfway(self): compiled = cuda.jit("void(float32[:], float32, int32)")(simple_round_to) ary = np.zeros(1, dtype=np.float32) # Value chosen to trigger the "round to even" branch of the # implementation val = 0.3425 ndigits = 3 compiled[1, 1](ary, val, ndigits) self.assertPreciseEqual(ary[0], round(val, ndigits), prec='single') def test_round_to_f8(self): compiled = cuda.jit("void(float64[:], float64, int32)")(simple_round_to) ary = np.zeros(1, dtype=np.float64) np.random.seed(123) vals = np.random.random(32) np.concatenate((vals, np.array([np.inf, -np.inf, np.nan]))) digits = (-5, -4, -3, -2, -1, 0, 1, 2, 3, 4, 5) for val, ndigits in itertools.product(vals, digits): with self.subTest(val=val, ndigits=ndigits): compiled[1, 1](ary, val, ndigits) self.assertPreciseEqual(ary[0], round(val, ndigits), prec='exact') # Trigger the "overflow safe" branch of the implementation val = 0.12345678987654321 * 10e-15 ndigits = 23 with self.subTest(val=val, ndigits=ndigits): compiled[1, 1](ary, val, ndigits) self.assertPreciseEqual(ary[0], round(val, ndigits), prec='double') # Skipped on cudasim for the same reasons as test_round_to_f4 above. @skip_on_cudasim('Overflow behavior differs on CPython') def test_round_to_f8_overflow(self): # Test that the input value is returned when y in round_ndigits # overflows. compiled = cuda.jit("void(float64[:], float64, int32)")(simple_round_to) ary = np.zeros(1, dtype=np.float64) val = np.finfo(np.float64).max # Unlike test_round_to_f4_overflow, a reasonable number of digits can # be used for this test to overflow y in round_ndigits. ndigits = 12 compiled[1, 1](ary, val, ndigits) self.assertEqual(ary[0], val) def test_round_to_f8_halfway(self): compiled = cuda.jit("void(float64[:], float64, int32)")(simple_round_to) ary = np.zeros(1, dtype=np.float64) # Value chosen to trigger the "round to even" branch of the # implementation, with a value that is not exactly representable with a # float32, but only a float64. val = 0.5425 ndigits = 3 compiled[1, 1](ary, val, ndigits) self.assertPreciseEqual(ary[0], round(val, ndigits), prec='double') if __name__ == '__main__': unittest.main()