From 764504df1e83a8c556ba68020376ac3cc5c57a68 Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Wed, 12 Feb 2025 16:29:33 +0100 Subject: [PATCH 1/7] Add a Tensor.repeat() method --- include/aidge/data/Spikegen.hpp | 10 +++++++++ include/aidge/data/Tensor.hpp | 16 +++++++++++--- src/data/Tensor.cpp | 37 +++++++++++++++++++++++++++++++++ 3 files changed, 60 insertions(+), 3 deletions(-) create mode 100644 include/aidge/data/Spikegen.hpp diff --git a/include/aidge/data/Spikegen.hpp b/include/aidge/data/Spikegen.hpp new file mode 100644 index 000000000..1267fda9d --- /dev/null +++ b/include/aidge/data/Spikegen.hpp @@ -0,0 +1,10 @@ +#ifndef AIDGE_CORE_DATA_SPIKEGEN_H_ +#define AIDGE_CORE_DATA_SPIKEGEN_H_ + +// Spikegen algorithm : +// +// time_data = data.repeat(time_steps) +// spike_data = rate_conv(time_data) +// return spike_data + +#endif diff --git a/include/aidge/data/Tensor.hpp b/include/aidge/data/Tensor.hpp index 5df59becd..76295270e 100644 --- a/include/aidge/data/Tensor.hpp +++ b/include/aidge/data/Tensor.hpp @@ -13,9 +13,7 @@ #define AIDGE_CORE_DATA_TENSOR_H_ #include <algorithm> -#include <cstddef> // std::size_t -#include <cstring> -#include <functional> // std::multiplies +#include <cstddef> // std::size_t #include <cstring> #include <functional> // std::multiplies #include <set> #include <memory> #include <numeric> // std::accumulate @@ -989,6 +987,18 @@ public: return ref(fallback, targetReqs.dataType(), device.first, device.second); } + + /** + * @brief Repeat the tensor along a new first dimension. + * For example, if the current tensor has dimensions (n, m), + * calling repeat(10) returns a tensor of shape (10, n, m) + * with 10 copies of the original data. + * + * @param times number of repetitions (must be positive) + * @return Tensor new tensor containing the repeated data. + */ + Tensor repeat(int times) const; + private: /** * @brief Compute the number of elements in the Tensor. diff --git a/src/data/Tensor.cpp b/src/data/Tensor.cpp index b128833c9..cf8430684 100644 --- a/src/data/Tensor.cpp +++ b/src/data/Tensor.cpp @@ -802,6 +802,43 @@ const Tensor& Tensor::ref(std::shared_ptr<Tensor>& fallback, } } +Tensor Tensor::repeat(int times) const { + AIDGE_ASSERT(times > 0, "repeat count must be positive"); + + // Ensure that the source tensor is contiguous. + Tensor src = *this; + if (not src.isContiguous()) { + src = src.clone(); + src.makeContiguous(); + } + + // Build new dimensions: new_dims = {times} followed by current dims. + std::vector<DimSize_t> newDims; + newDims.push_back(static_cast<DimSize_t>(times)); + for (const auto &d : dims()) { + newDims.push_back(d); + } + + // Create an output tensor with the new dimensions. + Tensor out(newDims); + out.setDataType(dataType(), false); + out.setDataFormat(dataFormat()); + if (hasImpl()) { + out.setBackend(getImpl()->backend(), device()); + } + + // Each "block" is a copy of the data from the original tensor. + const std::size_t block = src.size(); + // Loop over the repeat count and copy the block each time. + for (int i = 0; i < times; ++i) { + // out.getImpl()->copy(source pointer, number of elements, destination offset) + out.getImpl()->copy(src.getImpl()->rawPtr(src.getImplOffset()), + block, + i * block); + } + return out; +} + std::vector<std::size_t> Tensor::toCoord(const std::vector<DimSize_t>& dimensions, std::size_t index) { -- GitLab From 70a0be1f8b0949975cc3791294d589b610313715 Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Wed, 12 Feb 2025 16:41:55 +0100 Subject: [PATCH 2/7] Add unit test for Tensor.repeat() --- unit_tests/data/Test_Tensor.cpp | 29 +++++++++++++++++++++++++++++ 1 file changed, 29 insertions(+) diff --git a/unit_tests/data/Test_Tensor.cpp b/unit_tests/data/Test_Tensor.cpp index bfdc1a6b9..28f3a5fde 100644 --- a/unit_tests/data/Test_Tensor.cpp +++ b/unit_tests/data/Test_Tensor.cpp @@ -505,6 +505,35 @@ TEST_CASE("[core/data] Tensor(other)", "[Tensor][extract][zeros][print]") { } } + SECTION("repeat") { + Tensor tensor = Array2D<int, 2, 3>{{{1, 2, 3}, + {4, 5, 6}}}; + const int repeatTimes = 4; + + Tensor repeated; + REQUIRE_NOTHROW(repeated = tensor.repeat(repeatTimes)); + + // The expected shape after repeating is {repeatTimes, 2, 3} + std::vector<DimSize_t> expectedDims = {static_cast<DimSize_t>(repeatTimes), 2, 3}; + CHECK(repeated.dims() == expectedDims); + + // For each repetition along the new dimension, extract the slice and verify + // that it matches the original tensor + for (int i = 0; i < repeatTimes; ++i) { + Tensor slice; + REQUIRE_NOTHROW(slice = repeated.extract({static_cast<std::size_t>(i)})); + CHECK(slice.dims() == tensor.dims()); + + // Compare slice with original tensor elementwise + for (std::size_t idx = 0; idx < tensor.size(); ++idx) { + int expectedVal = tensor.get<int>(idx); + int sliceVal = slice.get<int>(idx); + INFO("Mismatch in repetition " << i << " at flat index " << idx); + CHECK(sliceVal == expectedVal); + } + } + } + // print, toString, SECTION("Pretty printing for debug") { Tensor x{}; -- GitLab From cada2936e4dd63a4c2a33a646dbe468e41e8b8eb Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Wed, 12 Feb 2025 17:52:40 +0100 Subject: [PATCH 3/7] First draft of spikegen rate convert function --- include/aidge/data/Spikegen.hpp | 21 +++++++++++---- src/data/SpikeGen.cpp | 45 +++++++++++++++++++++++++++++++++ 2 files changed, 61 insertions(+), 5 deletions(-) create mode 100644 src/data/SpikeGen.cpp diff --git a/include/aidge/data/Spikegen.hpp b/include/aidge/data/Spikegen.hpp index 1267fda9d..cbb2963f2 100644 --- a/include/aidge/data/Spikegen.hpp +++ b/include/aidge/data/Spikegen.hpp @@ -1,10 +1,21 @@ +/******************************************************************************** + * Copyright (c) 2025 CEA-List + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * SPDX-License-Identifier: EPL-2.0 + * + ********************************************************************************/ + #ifndef AIDGE_CORE_DATA_SPIKEGEN_H_ #define AIDGE_CORE_DATA_SPIKEGEN_H_ -// Spikegen algorithm : -// -// time_data = data.repeat(time_steps) -// spike_data = rate_conv(time_data) -// return spike_data +#include "aidge/data/Tensor.hpp" +namespace Aidge { +std::shared_ptr<Tensor> spikegenRate(std::shared_ptr<Tensor> tensor); +} + #endif diff --git a/src/data/SpikeGen.cpp b/src/data/SpikeGen.cpp new file mode 100644 index 000000000..977c64de9 --- /dev/null +++ b/src/data/SpikeGen.cpp @@ -0,0 +1,45 @@ +/******************************************************************************** + * Copyright (c) 2025 CEA-List + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * SPDX-License-Identifier: EPL-2.0 + * + ********************************************************************************/ + +#include <memory> +#include <random> + +#include "aidge/data/Spikegen.hpp" + +namespace Aidge { +std::shared_ptr<Tensor> rateConvert(const Tensor& tensor) { + + auto result = std::make_shared<Tensor>(tensor.clone()); + + // Bernoulli sampling + std::random_device rd; + std::mt19937 gen(rd()); + std::uniform_real_distribution<float> dis(0.0f, 1.0f); + + // Clip values between 0 and 1, equivalent to torch.clamp(min=0, max=1) + for (size_t i = 0; i < tensor.size(); i++) { + auto val = tensor.get<float>(i); + val = (val < 0.0f) ? 0.0f : ((val > 1.0f) ? 1.0f : val); + auto randomValue = dis(gen); + result->set(i, randomValue < val ? 1.0f : 0.0f); + } + + return result; +} + +std::shared_ptr<Tensor> spikegenRate(std::shared_ptr<Tensor> tensor) { + auto newTensor = tensor->repeat(10); + + newTensor.print(); // DEBUG + + return rateConvert(newTensor); +} +} // namespace Aidge -- GitLab From d12e93cd3aeaa540c9fc4c37c5eeec77410ebc78 Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Fri, 14 Feb 2025 10:09:35 +0100 Subject: [PATCH 4/7] Minor changes (doc, functions signatures) --- include/aidge/data/Spikegen.hpp | 12 +++++++++++- src/data/SpikeGen.cpp | 14 +++++--------- unit_tests/data/Test_Spikegen.cpp | 0 3 files changed, 16 insertions(+), 10 deletions(-) create mode 100644 unit_tests/data/Test_Spikegen.cpp diff --git a/include/aidge/data/Spikegen.hpp b/include/aidge/data/Spikegen.hpp index cbb2963f2..30fd5decb 100644 --- a/include/aidge/data/Spikegen.hpp +++ b/include/aidge/data/Spikegen.hpp @@ -12,9 +12,19 @@ #ifndef AIDGE_CORE_DATA_SPIKEGEN_H_ #define AIDGE_CORE_DATA_SPIKEGEN_H_ +#include <cstdint> + #include "aidge/data/Tensor.hpp" + namespace Aidge { -std::shared_ptr<Tensor> spikegenRate(std::shared_ptr<Tensor> tensor); + +/* + * @brief Spike rate encoding of input data + */ +Tensor spikegenRate(std::shared_ptr<Tensor> tensor, std::uint32_t numSteps); + + +Tensor spikegenLatency(std::shared_ptr<Tensor> tensor); } diff --git a/src/data/SpikeGen.cpp b/src/data/SpikeGen.cpp index 977c64de9..91b1ec433 100644 --- a/src/data/SpikeGen.cpp +++ b/src/data/SpikeGen.cpp @@ -15,11 +15,10 @@ #include "aidge/data/Spikegen.hpp" namespace Aidge { -std::shared_ptr<Tensor> rateConvert(const Tensor& tensor) { +Tensor rateConvert(const Tensor& tensor) { - auto result = std::make_shared<Tensor>(tensor.clone()); + auto result = tensor.clone(); - // Bernoulli sampling std::random_device rd; std::mt19937 gen(rd()); std::uniform_real_distribution<float> dis(0.0f, 1.0f); @@ -29,17 +28,14 @@ std::shared_ptr<Tensor> rateConvert(const Tensor& tensor) { auto val = tensor.get<float>(i); val = (val < 0.0f) ? 0.0f : ((val > 1.0f) ? 1.0f : val); auto randomValue = dis(gen); - result->set(i, randomValue < val ? 1.0f : 0.0f); + result.set(i, randomValue < val ? 1.0f : 0.0f); } return result; } -std::shared_ptr<Tensor> spikegenRate(std::shared_ptr<Tensor> tensor) { - auto newTensor = tensor->repeat(10); - - newTensor.print(); // DEBUG - +Tensor spikegenRate(std::shared_ptr<Tensor> tensor, std::uint32_t numSteps) { + auto newTensor = tensor->repeat(numSteps); return rateConvert(newTensor); } } // namespace Aidge diff --git a/unit_tests/data/Test_Spikegen.cpp b/unit_tests/data/Test_Spikegen.cpp new file mode 100644 index 000000000..e69de29bb -- GitLab From b76754c48cfc6cda5bc0ed8fc9b30711250c9950 Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Fri, 14 Feb 2025 11:11:02 +0100 Subject: [PATCH 5/7] Add basic test for Spikegen functions --- unit_tests/data/Test_Spikegen.cpp | 42 +++++++++++++++++++++++++++++++ 1 file changed, 42 insertions(+) diff --git a/unit_tests/data/Test_Spikegen.cpp b/unit_tests/data/Test_Spikegen.cpp index e69de29bb..06f33197d 100644 --- a/unit_tests/data/Test_Spikegen.cpp +++ b/unit_tests/data/Test_Spikegen.cpp @@ -0,0 +1,42 @@ +/******************************************************************************** + * Copyright (c) 2025 CEA-List + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * SPDX-License-Identifier: EPL-2.0 + * + ********************************************************************************/ + +#include <cstdint> // std::uint8_t, std::uint16_t, std::int32_t +#include <vector> // std::vector + +#include <catch2/catch_test_macros.hpp> + +#include "aidge/data/Spikegen.hpp" +#include "aidge/data/Tensor.hpp" +#include "aidge/utils/ArrayHelpers.hpp" +#include "aidge/utils/TensorUtils.hpp" + + +namespace Aidge +{ +TEST_CASE("[core/data] SpikeGen zeros", "[SpikeGen]") { + auto input = Tensor(Array1D<float, 3>({0,0,0})); + auto expectedOutput = Tensor(Array2D<float, 3, 3>({{{0,0,0}, {0,0,0}, {0,0,0}}})); + + auto spikes = spikegenRate(std::make_shared<Tensor>(input), 3); + + REQUIRE(approxEq<float>(spikes, expectedOutput)); +} + +TEST_CASE("[core/data] SpikeGen ones", "[SpikeGen]") { + auto input = Tensor(Array1D<float, 3>({1,1,1})); + auto expectedOutput = Tensor(Array2D<float, 3, 3>({{{1,1,1}, {1,1,1}, {1,1,1}}})); + + auto spikes = spikegenRate(std::make_shared<Tensor>(input), 3); + + REQUIRE(approxEq<float>(spikes, expectedOutput)); +} +} // namespace Aidge -- GitLab From 7bb394054dce714bca7e5aeffefc3219a4950333 Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Tue, 25 Feb 2025 10:32:38 +0100 Subject: [PATCH 6/7] Add python binding for spikegen rate function --- python_binding/data/pybind_Spikegen.cpp | 26 +++++++++++++++++++++++++ python_binding/pybind_core.cpp | 3 +++ 2 files changed, 29 insertions(+) create mode 100644 python_binding/data/pybind_Spikegen.cpp diff --git a/python_binding/data/pybind_Spikegen.cpp b/python_binding/data/pybind_Spikegen.cpp new file mode 100644 index 000000000..de4636bb2 --- /dev/null +++ b/python_binding/data/pybind_Spikegen.cpp @@ -0,0 +1,26 @@ +/******************************************************************************** + * Copyright (c) 2025 CEA-List + * + * This program and the accompanying materials are made available under + * the terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * SPDX-License-Identifier: EPL-2.0 + * + ********************************************************************************/ + +#include <pybind11/pybind11.h> + +#include "aidge/data/Spikegen.hpp" +#include "aidge/data/Tensor.hpp" + +namespace py = pybind11; +namespace Aidge { + +void init_Spikegen(py::module &m) { + m.def("spikegen_rate", &spikegenRate, + py::arg("tensor"), py::arg("numSteps"), + "Performs spike rate encoding on a Tensor"); +} + +} // namespace Aidge diff --git a/python_binding/pybind_core.cpp b/python_binding/pybind_core.cpp index ef1111b39..e9e246ad1 100644 --- a/python_binding/pybind_core.cpp +++ b/python_binding/pybind_core.cpp @@ -32,8 +32,10 @@ void init_OperatorImpl(py::module&); void init_Log(py::module&); void init_Operator(py::module&); void init_OperatorTensor(py::module&); +void init_Spikegen(py::module&); void init_StaticAnalysis(py::module&); + void init_Abs(py::module&); void init_Add(py::module&); void init_And(py::module&); @@ -122,6 +124,7 @@ void init_Aidge(py::module& m) { init_Tensor(m); init_TensorImpl(m); init_Attributes(m); + init_Spikegen(m); init_Node(m); init_GraphView(m); -- GitLab From fd3f8a3bd334fd3e57fd7a8f47826fcf130caabd Mon Sep 17 00:00:00 2001 From: Jerome Hue <jerome.hue@cea.fr> Date: Mon, 3 Mar 2025 10:10:45 +0100 Subject: [PATCH 7/7] Make a local function static Also remove some meaningless comments --- src/data/SpikeGen.cpp | 2 +- src/data/Tensor.cpp | 5 +---- 2 files changed, 2 insertions(+), 5 deletions(-) diff --git a/src/data/SpikeGen.cpp b/src/data/SpikeGen.cpp index 91b1ec433..eecd6c86e 100644 --- a/src/data/SpikeGen.cpp +++ b/src/data/SpikeGen.cpp @@ -15,7 +15,7 @@ #include "aidge/data/Spikegen.hpp" namespace Aidge { -Tensor rateConvert(const Tensor& tensor) { +static Tensor rateConvert(const Tensor& tensor) { auto result = tensor.clone(); diff --git a/src/data/Tensor.cpp b/src/data/Tensor.cpp index cf8430684..d44168564 100644 --- a/src/data/Tensor.cpp +++ b/src/data/Tensor.cpp @@ -805,21 +805,19 @@ const Tensor& Tensor::ref(std::shared_ptr<Tensor>& fallback, Tensor Tensor::repeat(int times) const { AIDGE_ASSERT(times > 0, "repeat count must be positive"); - // Ensure that the source tensor is contiguous. Tensor src = *this; if (not src.isContiguous()) { src = src.clone(); src.makeContiguous(); } - // Build new dimensions: new_dims = {times} followed by current dims. + // Build new dimensions: new_dims = {times} followed by current dims std::vector<DimSize_t> newDims; newDims.push_back(static_cast<DimSize_t>(times)); for (const auto &d : dims()) { newDims.push_back(d); } - // Create an output tensor with the new dimensions. Tensor out(newDims); out.setDataType(dataType(), false); out.setDataFormat(dataFormat()); @@ -829,7 +827,6 @@ Tensor Tensor::repeat(int times) const { // Each "block" is a copy of the data from the original tensor. const std::size_t block = src.size(); - // Loop over the repeat count and copy the block each time. for (int i = 0; i < times; ++i) { // out.getImpl()->copy(source pointer, number of elements, destination offset) out.getImpl()->copy(src.getImpl()->rawPtr(src.getImplOffset()), -- GitLab