From b10841184d5a7bd9ae3f0e387b4fc62a39006ca9 Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Wed, 14 Aug 2024 04:25:10 +0530 Subject: [PATCH 1/8] chore(bump cython): remove IF since support for usage of DEF/IF is expected to be removed, IF are remove from code as well. --- cyarray/carray.pyx | 7 ++----- cyarray/carray.pyx.mako | 8 +++----- 2 files changed, 5 insertions(+), 10 deletions(-) diff --git a/cyarray/carray.pyx b/cyarray/carray.pyx index 213ebc8..ec9698f 100644 --- a/cyarray/carray.pyx +++ b/cyarray/carray.pyx @@ -29,11 +29,8 @@ The numpy array may however be copied and used in any manner. # For malloc etc. from libc.stdlib cimport * -IF UNAME_SYSNAME == "Windows": - cdef extern from "msstdint.h" nogil: - ctypedef unsigned int uintptr_t -ELSE: - from libc.stdint cimport uintptr_t + +from libc.stdint cimport uintptr_t cimport numpy as np diff --git a/cyarray/carray.pyx.mako b/cyarray/carray.pyx.mako index 30208be..c6a3a91 100644 --- a/cyarray/carray.pyx.mako +++ b/cyarray/carray.pyx.mako @@ -1,4 +1,5 @@ <% +import platform type_info = [ ('int', 'IntArray', 'NPY_INT'), ('unsigned int', 'UIntArray', 'NPY_UINT'), @@ -37,11 +38,8 @@ The numpy array may however be copied and used in any manner. # For malloc etc. from libc.stdlib cimport * -IF UNAME_SYSNAME == "Windows": - cdef extern from "msstdint.h" nogil: - ctypedef unsigned int uintptr_t -ELSE: - from libc.stdint cimport uintptr_t + +from libc.stdint cimport uintptr_t cimport numpy as np From ef40b4b4bf0964b2bc1d5c9e4358aa9aa255dc86 Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Thu, 15 Aug 2024 12:17:46 +0530 Subject: [PATCH 2/8] cleanup: delete msstdint.h cyarray/msstdint.h was added in 24c49cc8446efeb2f9dd9a4e232ed12c095bcdba since stdint.h has not been there pre MSVC2015. It should be fine to remove this now. --- cyarray/msstdint.h | 259 --------------------------------------------- 1 file changed, 259 deletions(-) delete mode 100644 cyarray/msstdint.h diff --git a/cyarray/msstdint.h b/cyarray/msstdint.h deleted file mode 100644 index 4fe0ef9..0000000 --- a/cyarray/msstdint.h +++ /dev/null @@ -1,259 +0,0 @@ -// ISO C9x compliant stdint.h for Microsoft Visual Studio -// Based on ISO/IEC 9899:TC2 Committee draft (May 6, 2005) WG14/N1124 -// -// Copyright (c) 2006-2013 Alexander Chemeris -// -// 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. Neither the name of the product nor the names of its contributors may -// 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. -// -/////////////////////////////////////////////////////////////////////////////// - -#ifndef _MSC_VER // [ -#error "Use this header only with Microsoft Visual C++ compilers!" -#endif // _MSC_VER ] - -#ifndef _MSC_STDINT_H_ // [ -#define _MSC_STDINT_H_ - -#if _MSC_VER > 1000 -#pragma once -#endif - -#if _MSC_VER >= 1600 // [ -#include -#else // ] _MSC_VER >= 1600 [ - -#include - -// For Visual Studio 6 in C++ mode and for many Visual Studio versions when -// compiling for ARM we should wrap include with 'extern "C++" {}' -// or compiler give many errors like this: -// error C2733: second C linkage of overloaded function 'wmemchr' not allowed -#ifdef __cplusplus -extern "C" { -#endif -# include -#ifdef __cplusplus -} -#endif - -// Define _W64 macros to mark types changing their size, like intptr_t. -#ifndef _W64 -# if !defined(__midl) && (defined(_X86_) || defined(_M_IX86)) && _MSC_VER >= 1300 -# define _W64 __w64 -# else -# define _W64 -# endif -#endif - - -// 7.18.1 Integer types - -// 7.18.1.1 Exact-width integer types - -// Visual Studio 6 and Embedded Visual C++ 4 doesn't -// realize that, e.g. char has the same size as __int8 -// so we give up on __intX for them. -#if (_MSC_VER < 1300) - typedef signed char int8_t; - typedef signed short int16_t; - typedef signed int int32_t; - typedef unsigned char uint8_t; - typedef unsigned short uint16_t; - typedef unsigned int uint32_t; -#else - typedef signed __int8 int8_t; - typedef signed __int16 int16_t; - typedef signed __int32 int32_t; - typedef unsigned __int8 uint8_t; - typedef unsigned __int16 uint16_t; - typedef unsigned __int32 uint32_t; -#endif -typedef signed __int64 int64_t; -typedef unsigned __int64 uint64_t; - - -// 7.18.1.2 Minimum-width integer types -typedef int8_t int_least8_t; -typedef int16_t int_least16_t; -typedef int32_t int_least32_t; -typedef int64_t int_least64_t; -typedef uint8_t uint_least8_t; -typedef uint16_t uint_least16_t; -typedef uint32_t uint_least32_t; -typedef uint64_t uint_least64_t; - -// 7.18.1.3 Fastest minimum-width integer types -typedef int8_t int_fast8_t; -typedef int16_t int_fast16_t; -typedef int32_t int_fast32_t; -typedef int64_t int_fast64_t; -typedef uint8_t uint_fast8_t; -typedef uint16_t uint_fast16_t; -typedef uint32_t uint_fast32_t; -typedef uint64_t uint_fast64_t; - -// 7.18.1.4 Integer types capable of holding object pointers -#ifdef _WIN64 // [ - typedef signed __int64 intptr_t; - typedef unsigned __int64 uintptr_t; -#else // _WIN64 ][ - typedef _W64 signed int intptr_t; - typedef _W64 unsigned int uintptr_t; -#endif // _WIN64 ] - -// 7.18.1.5 Greatest-width integer types -typedef int64_t intmax_t; -typedef uint64_t uintmax_t; - - -// 7.18.2 Limits of specified-width integer types - -#if !defined(__cplusplus) || defined(__STDC_LIMIT_MACROS) // [ See footnote 220 at page 257 and footnote 221 at page 259 - -// 7.18.2.1 Limits of exact-width integer types -#define INT8_MIN ((int8_t)_I8_MIN) -#define INT8_MAX _I8_MAX -#define INT16_MIN ((int16_t)_I16_MIN) -#define INT16_MAX _I16_MAX -#define INT32_MIN ((int32_t)_I32_MIN) -#define INT32_MAX _I32_MAX -#define INT64_MIN ((int64_t)_I64_MIN) -#define INT64_MAX _I64_MAX -#define UINT8_MAX _UI8_MAX -#define UINT16_MAX _UI16_MAX -#define UINT32_MAX _UI32_MAX -#define UINT64_MAX _UI64_MAX - -// 7.18.2.2 Limits of minimum-width integer types -#define INT_LEAST8_MIN INT8_MIN -#define INT_LEAST8_MAX INT8_MAX -#define INT_LEAST16_MIN INT16_MIN -#define INT_LEAST16_MAX INT16_MAX -#define INT_LEAST32_MIN INT32_MIN -#define INT_LEAST32_MAX INT32_MAX -#define INT_LEAST64_MIN INT64_MIN -#define INT_LEAST64_MAX INT64_MAX -#define UINT_LEAST8_MAX UINT8_MAX -#define UINT_LEAST16_MAX UINT16_MAX -#define UINT_LEAST32_MAX UINT32_MAX -#define UINT_LEAST64_MAX UINT64_MAX - -// 7.18.2.3 Limits of fastest minimum-width integer types -#define INT_FAST8_MIN INT8_MIN -#define INT_FAST8_MAX INT8_MAX -#define INT_FAST16_MIN INT16_MIN -#define INT_FAST16_MAX INT16_MAX -#define INT_FAST32_MIN INT32_MIN -#define INT_FAST32_MAX INT32_MAX -#define INT_FAST64_MIN INT64_MIN -#define INT_FAST64_MAX INT64_MAX -#define UINT_FAST8_MAX UINT8_MAX -#define UINT_FAST16_MAX UINT16_MAX -#define UINT_FAST32_MAX UINT32_MAX -#define UINT_FAST64_MAX UINT64_MAX - -// 7.18.2.4 Limits of integer types capable of holding object pointers -#ifdef _WIN64 // [ -# define INTPTR_MIN INT64_MIN -# define INTPTR_MAX INT64_MAX -# define UINTPTR_MAX UINT64_MAX -#else // _WIN64 ][ -# define INTPTR_MIN INT32_MIN -# define INTPTR_MAX INT32_MAX -# define UINTPTR_MAX UINT32_MAX -#endif // _WIN64 ] - -// 7.18.2.5 Limits of greatest-width integer types -#define INTMAX_MIN INT64_MIN -#define INTMAX_MAX INT64_MAX -#define UINTMAX_MAX UINT64_MAX - -// 7.18.3 Limits of other integer types - -#ifdef _WIN64 // [ -# define PTRDIFF_MIN _I64_MIN -# define PTRDIFF_MAX _I64_MAX -#else // _WIN64 ][ -# define PTRDIFF_MIN _I32_MIN -# define PTRDIFF_MAX _I32_MAX -#endif // _WIN64 ] - -#define SIG_ATOMIC_MIN INT_MIN -#define SIG_ATOMIC_MAX INT_MAX - -#ifndef SIZE_MAX // [ -# ifdef _WIN64 // [ -# define SIZE_MAX _UI64_MAX -# else // _WIN64 ][ -# define SIZE_MAX _UI32_MAX -# endif // _WIN64 ] -#endif // SIZE_MAX ] - -// WCHAR_MIN and WCHAR_MAX are also defined in -#ifndef WCHAR_MIN // [ -# define WCHAR_MIN 0 -#endif // WCHAR_MIN ] -#ifndef WCHAR_MAX // [ -# define WCHAR_MAX _UI16_MAX -#endif // WCHAR_MAX ] - -#define WINT_MIN 0 -#define WINT_MAX _UI16_MAX - -#endif // __STDC_LIMIT_MACROS ] - - -// 7.18.4 Limits of other integer types - -#if !defined(__cplusplus) || defined(__STDC_CONSTANT_MACROS) // [ See footnote 224 at page 260 - -// 7.18.4.1 Macros for minimum-width integer constants - -#define INT8_C(val) val##i8 -#define INT16_C(val) val##i16 -#define INT32_C(val) val##i32 -#define INT64_C(val) val##i64 - -#define UINT8_C(val) val##ui8 -#define UINT16_C(val) val##ui16 -#define UINT32_C(val) val##ui32 -#define UINT64_C(val) val##ui64 - -// 7.18.4.2 Macros for greatest-width integer constants -// These #ifndef's are needed to prevent collisions with . -// Check out Issue 9 for the details. -#ifndef INTMAX_C // [ -# define INTMAX_C INT64_C -#endif // INTMAX_C ] -#ifndef UINTMAX_C // [ -# define UINTMAX_C UINT64_C -#endif // UINTMAX_C ] - -#endif // __STDC_CONSTANT_MACROS ] - -#endif // _MSC_VER >= 1600 ] - -#endif // _MSC_STDINT_H_ ] From 258e206da7a39ed07d3b215e47b292fe1f433c2d Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Tue, 13 Aug 2024 18:58:13 +0530 Subject: [PATCH 3/8] test(bechmarks): tests now include benchmarks --- cyarray/tests/test_carray.py | 345 ++++++++++++++++++++++++++++++++++- requirements.txt | 1 + setup.py | 2 +- 3 files changed, 346 insertions(+), 2 deletions(-) diff --git a/cyarray/tests/test_carray.py b/cyarray/tests/test_carray.py index 61f0bde..9c1f655 100644 --- a/cyarray/tests/test_carray.py +++ b/cyarray/tests/test_carray.py @@ -5,10 +5,10 @@ """ - # standard imports import unittest import numpy +import pytest # local imports from cyarray.carray import LongArray, py_aligned @@ -508,5 +508,348 @@ def test_reset_works_after_set_view(self): self.assertListEqual(view.get_npy_array().tolist(), expect) +class BenchmarkLongArray(unittest.TestCase): + """ + Tests for the LongArray class. + """ + + @pytest.fixture(autouse=True) + def setupBenchmark(self, benchmark): + self.benchmark = benchmark + + def test_constructor(self): + """ + Test the constructor. + """ + n = numpy.random.randint(low=10, high=100) + la = self.benchmark(LongArray, n) + + self.assertEqual(la.length, n) + self.assertEqual(la.alloc, n) + self.assertEqual(len(la.get_npy_array()), n) + + def test_set_indexing(self): + n = 100 + lab = LongArray(n) + self.benchmark.pedantic(lab.set, args=(9, n)) + self.assertEqual(lab[9], n) + + def test_get_indexing(self): + la = LongArray(100) + la[98] = 15 + res = self.benchmark(la.get, 98) + self.assertEqual(res, 15) + + def test_append(self): + lab = LongArray(0) + n = 100 + self.benchmark(lab.append, n) + self.assertEqual(lab[0], n) + + def test_reserve(self): + """ + Tests the reserve function. + """ + + def breserve(n): + la = LongArray(0) + la.reserve(n) + return la + + n = 100 + la = self.benchmark(breserve, n) + self.assertEqual(la.alloc, n) + + def test_resize(self): + """ + Tests the resize function. + """ + + def bresize(lab): + n = numpy.random.randint(low=10, high=20) + lab.resize(n) + return lab, n + + la = LongArray(10) + la, n = self.benchmark(bresize, la) + self.assertEqual(la.length, n) + + def test_get_npy_array(self): + la = LongArray(100) + la[0] = 1 + la[1] = 2 + la[2] = 3 + + nparray = self.benchmark(la.get_npy_array) + for i in range(3): + self.assertEqual(nparray[0], la[0]) + + def test_set_data(self): + """ + Tests the set_data function. + """ + n = 50 + la = LongArray(n) + np = numpy.arange(n) + self.benchmark(la.set_data, np) + + for i in range(n): + self.assertEqual(la[i], np[i]) + + self.assertRaises(ValueError, la.set_data, numpy.arange(55)) + + def test_squeeze(self): + + def bsqueeze(): + lab = LongArray(5) + lab.append(4) + lab.squeeze() + return lab + + la = self.benchmark(bsqueeze) + + self.assertEqual(la.length, 6) + self.assertEqual(la.alloc >= la.length, True) + self.assertEqual(len(la.get_npy_array()), 6) + + def test_reset(self): + def breset(): + lab = LongArray(5) + lab.reset() + return lab + + la = self.benchmark(breset) + + self.assertEqual(la.length, 0) + self.assertEqual(la.alloc, 5) + self.assertEqual(len(la.get_npy_array()), 0) + + def test_extend(self): + l2 = LongArray(5) + + for i in range(5): + l2[i] = 5 + i + + def bextend(l2n): + l1b = LongArray(0) + l1b.extend(l2n.get_npy_array()) + return l1b + + l1 = self.benchmark(bextend, l2) + + self.assertEqual(l1.length, 5) + self.assertEqual( + numpy.allclose( + l1.get_npy_array(), + numpy.arange(5, 10)), + True) + + def test_remove(self): + + def bremove(rem): + l1b = LongArray(10) + l1b.set_data(numpy.arange(10)) + l1b.remove(rem) + return l1b + + rem = [0, 4, 3] + l1 = self.benchmark(bremove, numpy.array(rem, dtype=int)) + + self.assertEqual(l1.length, 7) + self.assertEqual(numpy.allclose([7, 1, 2, 8, 9, 5, 6], + l1.get_npy_array()), True) + + def test_remove_with_strides(self): + + def bremove(rem): + l1b = LongArray(12) + l1b.set_data(numpy.arange(12)) + l1b.remove(rem, stride=3) + return l1b + + rem = [3, 1] + l1 = self.benchmark(bremove, numpy.array(rem, dtype=int)) + + # Then + self.assertEqual(l1.length, 6) + self.assertEqual(numpy.allclose([0, 1, 2, 6, 7, 8], + l1.get_npy_array()), True) + + # Given + l1 = LongArray(12) + l1.set_data(numpy.arange(12)) + + # When + rem = [0, 2] + l1.remove(numpy.array(rem, dtype=int), stride=3) + + # Then + self.assertEqual(l1.length, 6) + [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11] + self.assertEqual(numpy.allclose([9, 10, 11, 3, 4, 5], + l1.get_npy_array()), True) + + def test_align_array(self): + l1 = LongArray(10) + l1.set_data(numpy.arange(10)) + + new_indices = LongArray(10) + new_indices.set_data(numpy.asarray([1, 5, 3, 2, 4, 7, 8, 6, 9, 0])) + + l1.align_array(new_indices) + self.assertEqual(numpy.allclose([1, 5, 3, 2, 4, 7, 8, 6, 9, 0], + l1.get_npy_array()), True) + + # Test case with strides. + + def balign_array(): + l1b = LongArray(6) + l1b.set_data(numpy.arange(6)) + + new_indices = LongArray(3) + new_indices.set_data(numpy.asarray([2, 1, 0])) + l1b.align_array(new_indices, 2) + return l1b + + l1 = self.benchmark(balign_array) + self.assertEqual(numpy.allclose([4, 5, 2, 3, 0, 1], + l1.get_npy_array()), True) + + def test_copy_subset(self): + + def bcopy_subset(l2b): + l1b = LongArray(10) + l1b.set_data(numpy.arange(10)) + + # a valid copy. + l1b.copy_subset(l2b, 5, 9) + return l1b + + l2 = LongArray(4) + l2[0] = 4 + l2[1] = 3 + l2[2] = 2 + l2[3] = 1 + + l1 = self.benchmark(bcopy_subset, l2) + + self.assertEqual(numpy.allclose([0, 1, 2, 3, 4, 4, 3, 2, 1, 9], + l1.get_npy_array()), True) + + def test_copy_subset_works_with_strides(self): + def bcopy_subset(l2b): + l1b = LongArray(8) + l1b.set_data(numpy.arange(8)) + l1b.copy_subset(l2b, 2, 3, stride=2) + return l1b + + # Given + l2 = LongArray(4) + l2.set_data(numpy.arange(10, 14)) + + # When + l1 = self.benchmark(bcopy_subset, l2) + + # Then + numpy.testing.assert_array_equal( + l1.get_npy_array(), + [0, 1, 2, 3, 10, 11, 6, 7] + ) + + def test_copy_values(self): + def bcopy_values(l2b, indices): + l1b = LongArray(8) + l1b.set_data(numpy.arange(8)) + l1b.copy_values(indices, l2b) + return l1b + + # Given + l1 = LongArray(8) + l1.set_data(numpy.arange(8)) + l2 = LongArray(8) + l2.set_data(numpy.zeros(8, dtype=int)) + + # When + indices = LongArray(3) + indices.set_data(numpy.array([2, 4, 6])) + l1 = self.benchmark.pedantic(bcopy_values, args=(l2, indices)) + + # Then + numpy.testing.assert_array_equal( + l2.get_npy_array(), + [2, 4, 6] + [0] * 5 + ) + + def test_update_min_max(self): + """ + Tests the update_min_max function. + """ + + def bupdate_min_max(): + l1b = LongArray(10) + l1b.set_data(numpy.arange(10)) + l1b.update_min_max() + return l1b + + l1 = self.benchmark(bupdate_min_max) + + self.assertEqual(l1.minimum, 0) + self.assertEqual(l1.maximum, 9) + + def test_pickling(self): + """ + Tests the __reduce__ and __setstate__ functions. + """ + import pickle + + def bpickle(l1b): + l1_dump = pickle.dumps(l1b) + return pickle.loads(l1_dump) + + l1 = LongArray(3) + l1.set_data(numpy.arange(3)) + + l1_load = self.benchmark(bpickle, l1) + self.assertEqual( + (l1_load.get_npy_array() == l1.get_npy_array()).all(), True) + + def test_set_view(self): + # Given + src = LongArray() + src.extend(numpy.arange(5)) + + # When. + def bset_view(bsrc): + bview = LongArray() + bview.set_view(bsrc, 1, 4) + return bview + + view = self.benchmark(bset_view, src) + + # Then. + self.assertEqual(view.length, 3) + expect = list(range(1, 4)) + self.assertListEqual(view.get_npy_array().tolist(), expect) + + def test_set_view_for_empty_array(self): + # Given + src = LongArray() + src.extend(numpy.arange(5)) + + # When. + + def bset_view(bsrc): + view = LongArray() + view.set_view(bsrc, 1, 1) + return view + + view = self.benchmark(bset_view, src) + + # Then. + self.assertEqual(view.length, 0) + expect = [] + self.assertListEqual(view.get_npy_array().tolist(), expect) + + if __name__ == '__main__': unittest.main() diff --git a/requirements.txt b/requirements.txt index 64cd771..3b2a031 100644 --- a/requirements.txt +++ b/requirements.txt @@ -3,3 +3,4 @@ setuptools>=6.0 numpy mako pytest +pytest-benchmark[histogram] diff --git a/setup.py b/setup.py index fe1a5d2..bcf12ab 100644 --- a/setup.py +++ b/setup.py @@ -89,7 +89,7 @@ def setup_package(): install_requires = [ 'numpy', 'mako', 'Cython<3.0', 'setuptools>=6.0' ] - tests_require = ["pytest"] + tests_require = ["pytest", "pytest-benchmark[histogram]"] docs_require = ["sphinx"] ext_modules = get_basic_extensions() From b8d53b4c97920466565a7d394744886c1fb78323 Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Tue, 13 Aug 2024 18:58:37 +0530 Subject: [PATCH 4/8] ci(benchmarks): run test benchmarks on gh actions --- .github/workflows/tests.yml | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 0e4c992..6e26198 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -23,4 +23,13 @@ jobs: python -m pip install -r requirements.txt pip install . - name: Run tests - run: pytest -v --pyargs cyarray + run: pytest -v --pyargs cyarray --benchmark-save benchmark-stats --benchmark-json benchmark-full.json --benchmark-histogram + + - name: 'Upload Perf Data' + uses: actions/upload-artifact@v4 + with: + name: perf-bench-result-${{ matrix.python-version }}-${{ matrix.os }} + path: | + benchmark_*.svg + .benchmarks/* + benchmark-full.json From add5487f624d678a699c0c5c7d887d66d27669a1 Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Tue, 13 Aug 2024 19:21:11 +0530 Subject: [PATCH 5/8] ci: use setup.py instead of pip install --- .github/workflows/tests.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 6e26198..5dcbabc 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -21,7 +21,7 @@ jobs: run: | python -m pip install --upgrade pip setuptools wheel python -m pip install -r requirements.txt - pip install . + python setup.py develop - name: Run tests run: pytest -v --pyargs cyarray --benchmark-save benchmark-stats --benchmark-json benchmark-full.json --benchmark-histogram From 40094ee42ab1055a4c8707a5a3330fbed74d9441 Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Thu, 15 Aug 2024 12:45:38 +0530 Subject: [PATCH 6/8] chore(bump cython): fn c_reset fixed --- cyarray/carray.pxd | 2 +- cyarray/carray.pxd.mako | 2 +- cyarray/carray.pyx | 42 ++++++++++++++++++++++++++++++----------- cyarray/carray.pyx.mako | 10 +++++++--- 4 files changed, 40 insertions(+), 16 deletions(-) diff --git a/cyarray/carray.pxd b/cyarray/carray.pxd index bde925f..9706616 100644 --- a/cyarray/carray.pxd +++ b/cyarray/carray.pxd @@ -32,7 +32,7 @@ cdef class BaseArray: cdef void c_align_array(self, LongArray new_indices, int stride=*) nogil cdef void c_reserve(self, long size) nogil - cdef void c_reset(self) nogil + cdef void c_reset(self) cdef void c_resize(self, long size) nogil cdef void c_squeeze(self) nogil diff --git a/cyarray/carray.pxd.mako b/cyarray/carray.pxd.mako index 72f874c..a9ca9d0 100644 --- a/cyarray/carray.pxd.mako +++ b/cyarray/carray.pxd.mako @@ -40,7 +40,7 @@ cdef class BaseArray: cdef void c_align_array(self, LongArray new_indices, int stride=*) nogil cdef void c_reserve(self, long size) nogil - cdef void c_reset(self) nogil + cdef void c_reset(self) cdef void c_resize(self, long size) nogil cdef void c_squeeze(self) nogil diff --git a/cyarray/carray.pyx b/cyarray/carray.pyx index ec9698f..97e7d0f 100644 --- a/cyarray/carray.pyx +++ b/cyarray/carray.pyx @@ -149,7 +149,7 @@ cdef class BaseArray: cdef void c_reserve(self, long size) nogil: pass - cdef void c_reset(self) nogil: + cdef void c_reset(self): cdef PyArrayObject* arr = self._npy_array self.length = 0 arr.dimensions[0] = self.length @@ -443,12 +443,16 @@ cdef class IntArray(BaseArray): self.alloc = size arr.data = self.data - cdef void c_reset(self) nogil: + cdef void c_reset(self): + cdef np.npy_intp dims = self.length BaseArray.c_reset(self) if self._old_data != NULL: self.data = self._old_data self._old_data = NULL - self._npy_array.data = self.data + + self._npy_array = PyArray_SimpleNewFromData( + 1, &dims, NPY_INT, self.data + ) cdef void c_resize(self, long size) nogil: cdef PyArrayObject* arr = self._npy_array @@ -948,12 +952,16 @@ cdef class UIntArray(BaseArray): self.alloc = size arr.data = self.data - cdef void c_reset(self) nogil: + cdef void c_reset(self): + cdef np.npy_intp dims = self.length BaseArray.c_reset(self) if self._old_data != NULL: self.data = self._old_data self._old_data = NULL - self._npy_array.data = self.data + + self._npy_array = PyArray_SimpleNewFromData( + 1, &dims, NPY_UINT, self.data + ) cdef void c_resize(self, long size) nogil: cdef PyArrayObject* arr = self._npy_array @@ -1453,12 +1461,16 @@ cdef class LongArray(BaseArray): self.alloc = size arr.data = self.data - cdef void c_reset(self) nogil: + cdef void c_reset(self): + cdef np.npy_intp dims = self.length BaseArray.c_reset(self) if self._old_data != NULL: self.data = self._old_data self._old_data = NULL - self._npy_array.data = self.data + + self._npy_array = PyArray_SimpleNewFromData( + 1, &dims, NPY_LONG, self.data + ) cdef void c_resize(self, long size) nogil: cdef PyArrayObject* arr = self._npy_array @@ -1958,12 +1970,16 @@ cdef class FloatArray(BaseArray): self.alloc = size arr.data = self.data - cdef void c_reset(self) nogil: + cdef void c_reset(self): + cdef np.npy_intp dims = self.length BaseArray.c_reset(self) if self._old_data != NULL: self.data = self._old_data self._old_data = NULL - self._npy_array.data = self.data + + self._npy_array = PyArray_SimpleNewFromData( + 1, &dims, NPY_FLOAT, self.data + ) cdef void c_resize(self, long size) nogil: cdef PyArrayObject* arr = self._npy_array @@ -2463,12 +2479,16 @@ cdef class DoubleArray(BaseArray): self.alloc = size arr.data = self.data - cdef void c_reset(self) nogil: + cdef void c_reset(self): + cdef np.npy_intp dims = self.length BaseArray.c_reset(self) if self._old_data != NULL: self.data = self._old_data self._old_data = NULL - self._npy_array.data = self.data + + self._npy_array = PyArray_SimpleNewFromData( + 1, &dims, NPY_DOUBLE, self.data + ) cdef void c_resize(self, long size) nogil: cdef PyArrayObject* arr = self._npy_array diff --git a/cyarray/carray.pyx.mako b/cyarray/carray.pyx.mako index c6a3a91..b31f68f 100644 --- a/cyarray/carray.pyx.mako +++ b/cyarray/carray.pyx.mako @@ -159,7 +159,7 @@ cdef class BaseArray: cdef void c_reserve(self, long size) nogil: pass - cdef void c_reset(self) nogil: + cdef void c_reset(self): cdef PyArrayObject* arr = self._npy_array self.length = 0 arr.dimensions[0] = self.length @@ -456,12 +456,16 @@ cdef class ${CLASSNAME}(BaseArray): self.alloc = size arr.data = self.data - cdef void c_reset(self) nogil: + cdef void c_reset(self): + cdef np.npy_intp dims = self.length BaseArray.c_reset(self) if self._old_data != NULL: self.data = self._old_data self._old_data = NULL - self._npy_array.data = self.data + + self._npy_array = PyArray_SimpleNewFromData( + 1, &dims, ${NUMPY_TYPENAME}, self.data + ) cdef void c_resize(self, long size) nogil: cdef PyArrayObject* arr = self._npy_array From dedb698422b5869a3536cc3779a3304ce60776de Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Thu, 15 Aug 2024 12:47:44 +0530 Subject: [PATCH 7/8] fix(performance warnings): by adding noexcept --- cyarray/carray.pxd | 10 +++++----- cyarray/carray.pxd.mako | 10 +++++----- cyarray/carray.pyx | 20 ++++++++++---------- cyarray/carray.pyx.mako | 12 ++++++------ 4 files changed, 26 insertions(+), 26 deletions(-) diff --git a/cyarray/carray.pxd b/cyarray/carray.pxd index 9706616..5d22577 100644 --- a/cyarray/carray.pxd +++ b/cyarray/carray.pxd @@ -12,10 +12,10 @@ Declaration File. # numpy import cimport numpy as np -cdef long aligned(long n, int item_size) nogil -cdef void* aligned_malloc(size_t bytes) nogil -cdef void* aligned_realloc(void* existing, size_t bytes, size_t old_size) nogil -cdef void aligned_free(void* p) nogil +cdef long aligned(long n, int item_size) noexcept nogil +cdef void* aligned_malloc(size_t bytes) noexcept nogil +cdef void* aligned_realloc(void* existing, size_t bytes, size_t old_size) noexcept nogil +cdef void aligned_free(void* p) noexcept nogil # forward declaration cdef class BaseArray @@ -31,7 +31,7 @@ cdef class BaseArray: cdef np.ndarray _npy_array cdef void c_align_array(self, LongArray new_indices, int stride=*) nogil - cdef void c_reserve(self, long size) nogil + cdef void c_reserve(self, long size) noexcept nogil cdef void c_reset(self) cdef void c_resize(self, long size) nogil cdef void c_squeeze(self) nogil diff --git a/cyarray/carray.pxd.mako b/cyarray/carray.pxd.mako index a9ca9d0..da52cb1 100644 --- a/cyarray/carray.pxd.mako +++ b/cyarray/carray.pxd.mako @@ -20,10 +20,10 @@ Declaration File. # numpy import cimport numpy as np -cdef long aligned(long n, int item_size) nogil -cdef void* aligned_malloc(size_t bytes) nogil -cdef void* aligned_realloc(void* existing, size_t bytes, size_t old_size) nogil -cdef void aligned_free(void* p) nogil +cdef long aligned(long n, int item_size) noexcept nogil +cdef void* aligned_malloc(size_t bytes) noexcept nogil +cdef void* aligned_realloc(void* existing, size_t bytes, size_t old_size) noexcept nogil +cdef void aligned_free(void* p) noexcept nogil # forward declaration cdef class BaseArray @@ -39,7 +39,7 @@ cdef class BaseArray: cdef np.ndarray _npy_array cdef void c_align_array(self, LongArray new_indices, int stride=*) nogil - cdef void c_reserve(self, long size) nogil + cdef void c_reserve(self, long size) noexcept nogil cdef void c_reset(self) cdef void c_resize(self, long size) nogil cdef void c_squeeze(self) nogil diff --git a/cyarray/carray.pyx b/cyarray/carray.pyx index 97e7d0f..4f90721 100644 --- a/cyarray/carray.pyx +++ b/cyarray/carray.pyx @@ -65,7 +65,7 @@ cdef extern from "stdlib.h": # numpy module initialization call _import_array() -cdef inline long aligned(long n, int item_size) nogil: +cdef inline long aligned(long n, int item_size) noexcept nogil: """Align `n` items each having size (in bytes) `item_size` to 64 bytes and return the appropriate number of items that would be aligned to 64 bytes. @@ -124,13 +124,13 @@ cdef void* _deref_base(void* ptr) nogil: raise MemoryError("Passed pointer is not aligned.") return base -cdef void* aligned_malloc(size_t bytes) nogil: +cdef void* aligned_malloc(size_t bytes) noexcept nogil: return _aligned_malloc(bytes) -cdef void* aligned_realloc(void* p, size_t bytes, size_t old_size) nogil: +cdef void* aligned_realloc(void* p, size_t bytes, size_t old_size) noexcept nogil: return _aligned_realloc(p, bytes, old_size) -cdef void aligned_free(void* p) nogil: +cdef void aligned_free(void* p) noexcept nogil: """Free block allocated by alligned_malloc. """ free(_deref_base(p)) @@ -146,7 +146,7 @@ cdef class BaseArray: """ pass - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: pass cdef void c_reset(self): @@ -425,7 +425,7 @@ cdef class IntArray(BaseArray): # update the numpy arrays length arr.dimensions[0] = self.length - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: cdef PyArrayObject* arr = self._npy_array cdef void* data = NULL if size > self.alloc: @@ -934,7 +934,7 @@ cdef class UIntArray(BaseArray): # update the numpy arrays length arr.dimensions[0] = self.length - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: cdef PyArrayObject* arr = self._npy_array cdef void* data = NULL if size > self.alloc: @@ -1443,7 +1443,7 @@ cdef class LongArray(BaseArray): # update the numpy arrays length arr.dimensions[0] = self.length - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: cdef PyArrayObject* arr = self._npy_array cdef void* data = NULL if size > self.alloc: @@ -1952,7 +1952,7 @@ cdef class FloatArray(BaseArray): # update the numpy arrays length arr.dimensions[0] = self.length - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: cdef PyArrayObject* arr = self._npy_array cdef void* data = NULL if size > self.alloc: @@ -2461,7 +2461,7 @@ cdef class DoubleArray(BaseArray): # update the numpy arrays length arr.dimensions[0] = self.length - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: cdef PyArrayObject* arr = self._npy_array cdef void* data = NULL if size > self.alloc: diff --git a/cyarray/carray.pyx.mako b/cyarray/carray.pyx.mako index b31f68f..d254a75 100644 --- a/cyarray/carray.pyx.mako +++ b/cyarray/carray.pyx.mako @@ -74,7 +74,7 @@ cdef extern from "stdlib.h": # numpy module initialization call _import_array() -cdef inline long aligned(long n, int item_size) nogil: +cdef inline long aligned(long n, int item_size) noexcept nogil: """Align `n` items each having size (in bytes) `item_size` to 64 bytes and return the appropriate number of items that would be aligned to 64 bytes. @@ -133,13 +133,13 @@ cdef void* _deref_base(void* ptr) nogil: raise MemoryError("Passed pointer is not aligned.") return base -cdef void* aligned_malloc(size_t bytes) nogil: +cdef void* aligned_malloc(size_t bytes) noexcept nogil: return _aligned_malloc(bytes) -cdef void* aligned_realloc(void* p, size_t bytes, size_t old_size) nogil: +cdef void* aligned_realloc(void* p, size_t bytes, size_t old_size) noexcept nogil: return _aligned_realloc(p, bytes, old_size) -cdef void aligned_free(void* p) nogil: +cdef void aligned_free(void* p) noexcept nogil: """Free block allocated by alligned_malloc. """ free(_deref_base(p)) @@ -156,7 +156,7 @@ cdef class BaseArray: """ pass - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: pass cdef void c_reset(self): @@ -438,7 +438,7 @@ cdef class ${CLASSNAME}(BaseArray): # update the numpy arrays length arr.dimensions[0] = self.length - cdef void c_reserve(self, long size) nogil: + cdef void c_reserve(self, long size) noexcept nogil: cdef PyArrayObject* arr = self._npy_array cdef void* data = NULL if size > self.alloc: From b24698de24a3791027a04ffdfcf52687a7f5aa43 Mon Sep 17 00:00:00 2001 From: Navaneet Villodi <11260095+nauaneed@users.noreply.github.com> Date: Wed, 14 Aug 2024 14:57:08 +0530 Subject: [PATCH 8/8] chore(bump cython): no more pinned 'cython<3.0' --- pyproject.toml | 2 +- requirements.txt | 2 +- setup.py | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index b53d030..6eb1fc9 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -3,6 +3,6 @@ requires = [ "wheel>=0.29.0", "setuptools>=42.0.0", "oldest-supported-numpy", - "Cython<3.0", + "Cython", "mako" ] \ No newline at end of file diff --git a/requirements.txt b/requirements.txt index 3b2a031..0926b8a 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,4 +1,4 @@ -Cython<3.0 +Cython setuptools>=6.0 numpy mako diff --git a/setup.py b/setup.py index bcf12ab..3801e83 100644 --- a/setup.py +++ b/setup.py @@ -87,7 +87,7 @@ def setup_package(): # The requirements. install_requires = [ - 'numpy', 'mako', 'Cython<3.0', 'setuptools>=6.0' + 'numpy', 'mako', 'Cython', 'setuptools>=6.0' ] tests_require = ["pytest", "pytest-benchmark[histogram]"] docs_require = ["sphinx"]