From 21796aba074d0a684a8aab9ad3ddeecbb5402bdf Mon Sep 17 00:00:00 2001 From: David Rotermund <54365609+davrot@users.noreply.github.com> Date: Sat, 4 Feb 2023 14:27:36 +0100 Subject: [PATCH] Add files via upload --- .../h_dynamic_cnn_cpu_cpp/HDynamicCNNCPU.cpp | 40 +- .../HDynamicCNNGPU.cu | 542 +++++++++++++ .../h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.h | 84 ++ network/h_dynamic_cnn_gpu_cpp_v1/Makefile | 33 + .../PyHDynamicCNNGPU.cpp | 18 + .../HDynamicCNNGPU.cu | 727 ++++++++++++++++++ .../HDynamicCNNGPU.h | 113 +++ network/h_dynamic_cnn_gpu_cpp_v2_pre/Makefile | 153 ++++ .../PyHDynamicCNNGPU.cpp | 18 + .../kernel_helper_functions.cu | 17 + .../kernel_helper_functions.h | 7 + .../kernel_phxy_fill_with_h.cu | 64 ++ .../kernel_phxy_fill_with_h.h | 18 + .../kernel_phxy_fill_with_spike_selected_w.cu | 73 ++ .../kernel_phxy_fill_with_spike_selected_w.h | 17 + .../kernel_phxy_one_over_sum_into_pxy.cu | 74 ++ .../kernel_phxy_one_over_sum_into_pxy.h | 17 + .../kernel_phxy_plus_phxy.cu | 51 ++ .../kernel_phxy_plus_phxy.h | 12 + .../kernel_phxy_plus_pxy.cu | 70 ++ .../kernel_phxy_plus_pxy.h | 16 + .../kernel_phxy_times_phxy_equals_phxy.cu | 53 ++ .../kernel_phxy_times_phxy_equals_phxy.h | 15 + .../kernel_phxy_times_pxy.cu | 70 ++ .../kernel_phxy_times_pxy.h | 17 + .../kernel_pxy_plus_v.cu | 50 ++ .../kernel_pxy_plus_v.h | 12 + .../kernel_pxy_reciprocal.cu | 50 ++ .../kernel_pxy_reciprocal.h | 12 + .../kernel_pxy_set_to_v.cu | 50 ++ .../kernel_pxy_set_to_v.h | 13 + .../kernel_pxy_time_pxy.cu | 58 ++ .../kernel_pxy_time_pxy.h | 15 + .../kernel_pxy_times_spike_selected_sxy.cu | 73 ++ .../kernel_pxy_times_spike_selected_sxy.h | 18 + .../kernel_pxy_times_v.cu | 50 ++ .../kernel_pxy_times_v.h | 13 + .../o/HDynamicCNNGPU.o | Bin 0 -> 33080 bytes .../o/PyHDynamicCNNGPU.o | Bin 0 -> 367368 bytes .../o/kernel_helper_functions.o | Bin 0 -> 7840 bytes .../o/kernel_phxy_fill_with_h.o | Bin 0 -> 21000 bytes .../kernel_phxy_fill_with_spike_selected_w.o | Bin 0 -> 24816 bytes .../o/kernel_phxy_one_over_sum_into_pxy.o | Bin 0 -> 26520 bytes .../o/kernel_phxy_plus_phxy.o | Bin 0 -> 14744 bytes .../o/kernel_phxy_plus_pxy.o | Bin 0 -> 22008 bytes .../o/kernel_phxy_times_phxy_equals_phxy.o | Bin 0 -> 15352 bytes .../o/kernel_phxy_times_pxy.o | Bin 0 -> 22040 bytes .../o/kernel_pxy_plus_v.o | Bin 0 -> 14272 bytes .../o/kernel_pxy_reciprocal.o | Bin 0 -> 15680 bytes .../o/kernel_pxy_set_to_v.o | Bin 0 -> 14264 bytes .../o/kernel_pxy_time_pxy.o | Bin 0 -> 14584 bytes .../o/kernel_pxy_times_spike_selected_sxy.o | Bin 0 -> 23088 bytes .../o/kernel_pxy_times_v.o | Bin 0 -> 14280 bytes network/h_dynamic_cnn_gpu_cpp_v2_pre/test.sh | 14 + 54 files changed, 2735 insertions(+), 12 deletions(-) create mode 100644 network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v1/Makefile create mode 100644 network/h_dynamic_cnn_gpu_cpp_v1/PyHDynamicCNNGPU.cpp create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/Makefile create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/PyHDynamicCNNGPU.cpp create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.cu create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.h create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/HDynamicCNNGPU.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/PyHDynamicCNNGPU.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_helper_functions.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_fill_with_h.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_fill_with_spike_selected_w.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_one_over_sum_into_pxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_plus_phxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_plus_pxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_times_phxy_equals_phxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_times_pxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_plus_v.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_reciprocal.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_set_to_v.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_time_pxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_times_spike_selected_sxy.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_times_v.o create mode 100644 network/h_dynamic_cnn_gpu_cpp_v2_pre/test.sh diff --git a/network/h_dynamic_cnn_cpu_cpp/HDynamicCNNCPU.cpp b/network/h_dynamic_cnn_cpu_cpp/HDynamicCNNCPU.cpp index 3722929..af30f32 100644 --- a/network/h_dynamic_cnn_cpu_cpp/HDynamicCNNCPU.cpp +++ b/network/h_dynamic_cnn_cpu_cpp/HDynamicCNNCPU.cpp @@ -64,13 +64,20 @@ void HDynamicCNNCPU::entrypoint( size_t h_dim_c1 = h_dim_2 * h_dim_3; size_t h_dim_c2 = h_dim_3; - float* epsilon_xy_pointer = (float*)epsilon_xy_pointer_addr; - assert((epsilon_xy_pointer != nullptr)); - assert((epsilon_xy_dim_0 > 0)); - assert((epsilon_xy_dim_1 > 0)); + float* epsilon_xy_pointer = nullptr; + size_t epsilon_xy_dim_c0 = 0; + size_t epsilon_xy_dim_c1 = 0; + if (epsilon_xy_pointer_addr != 0) + { + epsilon_xy_pointer = (float*)epsilon_xy_pointer_addr; + assert((epsilon_xy_pointer != nullptr)); + assert((epsilon_xy_dim_0 > 0)); + assert((epsilon_xy_dim_1 > 0)); + assert((epsilon_xy_dim_2 > 0)); - size_t epsilon_xy_dim_c0 = epsilon_xy_dim_2 * epsilon_xy_dim_1; - size_t epsilon_xy_dim_c1 = epsilon_xy_dim_2; + epsilon_xy_dim_c0 = epsilon_xy_dim_2 * epsilon_xy_dim_1; + epsilon_xy_dim_c1 = epsilon_xy_dim_2; + } float* epsilon_t_pointer = (float*)epsilon_t_pointer_addr; assert((epsilon_t_pointer != nullptr)); @@ -164,16 +171,18 @@ void HDynamicCNNCPU::update( { float* h_ptr; - float* epsilon_xy_ptr; + float* epsilon_xy_ptr = nullptr; int64_t* input_ptr; for (size_t counter_x = 0; counter_x < dim_x; counter_x++) { for (size_t counter_y = 0; counter_y < dim_y; counter_y++) { - epsilon_xy_ptr = epsilon_xy_pointer + - counter_x * epsilon_xy_dim_c1 + counter_y; - + if (epsilon_xy_dim_c1 != 0) + { + epsilon_xy_ptr = epsilon_xy_pointer + + counter_x * epsilon_xy_dim_c1 + counter_y; + } h_ptr = h_pointer + pattern_id * h_dim_c0 + counter_x * h_dim_c2 + counter_y; @@ -252,8 +261,15 @@ void HDynamicCNNCPU::update_one_ip( if (*spike >= 0) { - epsilon_subsegment = - epsilon_xy_pointer[*spike * epsilon_xy_dim_c0] * epsilon_t_pointer[counter_spike]; + if (epsilon_xy_dim_c0 != 0) + { + epsilon_subsegment = + epsilon_xy_pointer[*spike * epsilon_xy_dim_c0] * epsilon_t_pointer[counter_spike]; + } + else + { + epsilon_subsegment = epsilon_t_pointer[counter_spike]; + } w_ptr = weights_pointer + *spike * weights_dim_c0; diff --git a/network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.cu b/network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.cu new file mode 100644 index 0000000..0eb402f --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.cu @@ -0,0 +1,542 @@ +#include "HDynamicCNNGPU.h" + +#include +#include +#include + +#include +#include +#include + + +HDynamicCNNGPU::HDynamicCNNGPU() +{ + +}; + +HDynamicCNNGPU::~HDynamicCNNGPU() +{ + +}; + +void HDynamicCNNGPU::entrypoint( + int64_t h_pointer_addr, + int64_t h_dim_0, + int64_t h_dim_1, + int64_t h_dim_2, + int64_t h_dim_3, + int64_t epsilon_xy_pointer_addr, + int64_t epsilon_xy_dim_0, + int64_t epsilon_xy_dim_1, + int64_t epsilon_xy_dim_2, + int64_t epsilon_t_pointer_addr, + int64_t epsilon_t_dim_0, + int64_t weights_pointer_addr, + int64_t weights_dim_0, + int64_t weights_dim_1, + int64_t input_pointer_addr, + int64_t input_dim_0, + int64_t input_dim_1, + int64_t input_dim_2, + int64_t input_dim_3, + int64_t init_vector_pointer_addr, + int64_t init_vector_dim_0, + int64_t number_of_processes, + float forgetting_offset, + int64_t gpu_tuning_factor) +{ + + size_t number_of_pattern = input_dim_0; + + size_t h_dim = init_vector_dim_0; + float* h_init_ptr = (float*)init_vector_pointer_addr; + assert((h_init_ptr != nullptr)); + assert((h_dim > 0)); + + float* h_pointer = (float*)h_pointer_addr; + assert((h_pointer != nullptr)); + assert((h_dim_0 > 0)); + assert((h_dim_1 > 0)); + assert((h_dim_2 > 0)); + assert((h_dim_3 > 0)); + + size_t h_dim_c0 = h_dim_1 * h_dim_2 * h_dim_3; + size_t h_dim_c1 = h_dim_2 * h_dim_3; + size_t h_dim_c2 = h_dim_3; + + float* epsilon_xy_pointer = nullptr; + size_t epsilon_xy_dim_c0 = 0; + size_t epsilon_xy_dim_c1 = 0; + if (epsilon_xy_pointer_addr != 0) + { + epsilon_xy_pointer = (float*)epsilon_xy_pointer_addr; + assert((epsilon_xy_pointer != nullptr)); + assert((epsilon_xy_dim_0 > 0)); + assert((epsilon_xy_dim_1 > 0)); + assert((epsilon_xy_dim_2 > 0)); + + epsilon_xy_dim_c0 = epsilon_xy_dim_2 * epsilon_xy_dim_1; + epsilon_xy_dim_c1 = epsilon_xy_dim_2; + } + + float* epsilon_t_pointer = (float*)epsilon_t_pointer_addr; + assert((epsilon_t_pointer != nullptr)); + assert((epsilon_t_dim_0 > 0)); + + float* weights_pointer = (float*)weights_pointer_addr; + assert((weights_pointer != nullptr)); + assert((weights_dim_0 > 0)); + assert((weights_dim_1 > 0)); + + size_t weights_dim_c0 = weights_dim_1; + + int64_t* input_pointer = (int64_t*)input_pointer_addr; + assert((input_pointer != nullptr)); + assert((input_dim_0 > 0)); + assert((input_dim_1 > 0)); + assert((input_dim_2 > 0)); + assert((input_dim_3 > 0)); + + size_t input_dim_c0 = input_dim_1 * input_dim_2 * input_dim_3; + size_t input_dim_c1 = input_dim_2 * input_dim_3; + size_t input_dim_c2 = input_dim_3; + + assert((h_dim == weights_dim_1)); + size_t number_of_spikes = input_dim_1; + size_t dim_x = input_dim_2; + size_t dim_y = input_dim_3; + + float forgetting_offset_local = forgetting_offset / static_cast(h_dim); + + // -------------------- + assert((number_of_processes <= 0)); + + gpu_update( + h_init_ptr, + h_pointer, + h_dim_c0, + h_dim_c1, + h_dim_c2, + h_dim, + epsilon_xy_pointer, + epsilon_xy_dim_c0, + epsilon_xy_dim_c1, + epsilon_t_pointer, + weights_pointer, + weights_dim_c0, + input_pointer, + input_dim_c0, + input_dim_c1, + input_dim_c2, + number_of_spikes, + dim_x, + dim_y, + forgetting_offset, + forgetting_offset_local, + number_of_pattern, + gpu_tuning_factor); + + return; +}; + +__device__ void gpu_update_one_ip( + float* __restrict__ h_init_ptr, + float* __restrict__ h_pointer, + size_t h_dim_c1, + size_t h_dim, + float* __restrict__ weights_pointer, + size_t weights_dim_c0, + int64_t* input_pointer, + size_t input_dim_c1, + float* __restrict__ epsilon_xy_pointer, + size_t epsilon_xy_dim_c0, + float* __restrict__ epsilon_t_pointer, + size_t number_of_spikes, + float forgetting_offset, + float forgetting_offset_local, + float* __restrict__ h_temp, + float* __restrict__ h_subsegment +) +{ + + float h_temp_sum; + float temp_value; + + float epsilon_subsegment; + float epsilon_scale = 1.0; + + int64_t* spike; + float* w_ptr; + + // float* h_temp = new float[h_dim]; + // float* h_subsegment = new float[h_dim]; + + // Initialize the sub-segement + for (size_t counter = 0; counter < h_dim; counter++) + { + h_subsegment[counter] = h_init_ptr[counter]; + } + + for (size_t counter_spike = 0; counter_spike < number_of_spikes; counter_spike++) + { + if (epsilon_scale > 1E10) + { + temp_value = 1.0 / epsilon_scale; + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_subsegment[counter] *= temp_value; + } + + epsilon_scale = 1.0; + } + + spike = input_pointer + counter_spike * input_dim_c1; + + if (*spike >= 0) + { + if (epsilon_xy_dim_c0 != 0) + { + epsilon_subsegment = + epsilon_xy_pointer[*spike *epsilon_xy_dim_c0] * epsilon_t_pointer[counter_spike]; + } + else + { + epsilon_subsegment = epsilon_t_pointer[counter_spike]; + } + w_ptr = weights_pointer + *spike * weights_dim_c0; + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_temp[counter] = h_subsegment[counter] * w_ptr[counter]; + } + + h_temp_sum = 0.0; + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_temp_sum += h_temp[counter]; + } + + if (h_temp_sum > 1E-10) + { + temp_value = epsilon_scale * epsilon_subsegment / h_temp_sum; + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_temp[counter] *= temp_value; + } + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_subsegment[counter] += h_temp[counter]; + } + + if (forgetting_offset_local > 0.0) + { + temp_value = + epsilon_scale * epsilon_subsegment * forgetting_offset_local; + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_subsegment[counter] += temp_value; + } + + epsilon_scale *= + 1.0 + epsilon_subsegment * (1.0 + forgetting_offset); + } + else + { + epsilon_scale *= 1.0 + epsilon_subsegment * 1.0; + } + } + } + } + + temp_value = 1.0 / epsilon_scale; + + for (size_t counter = 0; counter < h_dim; counter++) + { + h_pointer[counter * h_dim_c1] = + h_subsegment[counter] * temp_value; + } + + // delete[] h_temp; + // delete[] h_subsegment; + + return; +}; + +__global__ void kernel_spike_generation( + float* __restrict__ h_init_ptr, + float* __restrict__ h_pointer, + size_t h_dim_c0, + size_t h_dim_c1, + size_t h_dim_c2, + size_t h_dim, + float* __restrict__ weights_pointer, + size_t weights_dim_c0, + int64_t* __restrict__ input_pointer, + size_t input_dim_c0, + size_t input_dim_c1, + size_t input_dim_c2, + float* __restrict__ epsilon_xy_pointer, + size_t epsilon_xy_dim_c0, + size_t epsilon_xy_dim_c1, + float* __restrict__ epsilon_t_pointer, + size_t number_of_spikes, + float forgetting_offset, + float forgetting_offset_local, + size_t dim_x, + size_t dim_y, + size_t dim_xy, + size_t max_threadable_tasks, + float* __restrict__ temp_memory_a, + float* __restrict__ temp_memory_b +) +{ + + int idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_threadable_tasks) + { + float* h_ptr; + float* epsilon_xy_ptr = nullptr; + int64_t* input_ptr; + + float* temp_memory_ptr_a = temp_memory_a + idx * h_dim; + float* temp_memory_ptr_b = temp_memory_b + idx * h_dim; + + // int pattern_id = idx; + int pattern_id = idx / dim_xy; + int position_xy = idx - (pattern_id * dim_xy); + + // size_t position_x = blockIdx.y; + // size_t position_y = blockIdx.z; + size_t position_x = position_xy / dim_y; + size_t position_y = position_xy - (position_x * dim_y); + + if (epsilon_xy_dim_c1 != 0) + { + epsilon_xy_ptr = epsilon_xy_pointer + + position_x * epsilon_xy_dim_c1 + position_y; + } + + h_ptr = h_pointer + + pattern_id * h_dim_c0 + position_x * h_dim_c2 + position_y; + + input_ptr = input_pointer + + pattern_id * input_dim_c0 + position_x * input_dim_c2 + position_y; + + gpu_update_one_ip( + h_init_ptr, + h_ptr, + h_dim_c1, + h_dim, + weights_pointer, + weights_dim_c0, + input_ptr, + input_dim_c1, + epsilon_xy_ptr, + epsilon_xy_dim_c0, + epsilon_t_pointer, + number_of_spikes, + forgetting_offset, + forgetting_offset_local, + temp_memory_ptr_a, + temp_memory_ptr_b + ); + + } + +}; + +// Let's face it... We need a better way to paralelize it... +void HDynamicCNNGPU::gpu_update( + float* h_init_ptr, + float* h_pointer, + size_t h_dim_c0, + size_t h_dim_c1, + size_t h_dim_c2, + size_t h_dim, + float* epsilon_xy_pointer, + size_t epsilon_xy_dim_c0, + size_t epsilon_xy_dim_c1, + float* epsilon_t_pointer, + float* weights_pointer, + size_t weights_dim_c0, + int64_t* input_pointer, + size_t input_dim_c0, + size_t input_dim_c1, + size_t input_dim_c2, + size_t number_of_spikes, + size_t dim_x, + size_t dim_y, + float forgetting_offset, + float forgetting_offset_local, + size_t number_of_pattern, + size_t gpu_tuning_factor) +{ + + cudaError_t status; + assert((dim_x < 65535)); + assert((dim_y < 65535)); + + // ////////////////////////////////////// + // Calculate the distribution on the GPU + // ////////////////////////////////////// + + int min_grid_size; + int block_size; + int grid_size; + + size_t dynamic_s_mem_size = 0; + size_t max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + // https://docs.nvidia.com/cuda/cuda-c-programming-guide/index.html?highlight=blocksize#occupancy-calculator + status = cudaOccupancyMaxPotentialBlockSize(&min_grid_size, &block_size, + (void*)kernel_spike_generation, + dynamic_s_mem_size, max_threadable_tasks); + if (status != cudaSuccess) + { + std::cerr << "CUDA Runtime Error at: " + << __FILE__ + << ":" + << __LINE__ + << std::endl; + std::cerr << cudaGetErrorString(status) << std::endl; + } + assert((status == cudaSuccess)); + + // https://docs.nvidia.com/cuda/cuda-c-programming-guide/index.html#features-and-technical-specifications + // Maximum dimensionality of grid of thread blocks: 3 + // Maximum x -dimension of a grid of thread blocks: (2^31)-1 + // Maximum y- or z-dimension of a grid of thread blocks: 65535 + + // Reduce the automatic block size with our guess + if ((gpu_tuning_factor > 0) && (gpu_tuning_factor < block_size)) + { + block_size = int(gpu_tuning_factor); + } + // Round up according to array size + // (I will separate x and y into other grid dimentsions soon) + // grid_size = (number_of_pattern + block_size - 1) / block_size; + grid_size = (max_threadable_tasks + block_size - 1) / block_size; + + float* temp_memory_a = nullptr; + status = cudaMalloc((void**)&temp_memory_a, h_dim * max_threadable_tasks * sizeof(float)); + if (status != cudaSuccess) + { + std::cerr << "CUDA Runtime Error at: " + << __FILE__ + << ":" + << __LINE__ + << std::endl; + std::cerr << cudaGetErrorString(status) << std::endl; + } + assert((status == cudaSuccess)); + + float* temp_memory_b = nullptr; + status = cudaMalloc((void**)&temp_memory_b, h_dim * max_threadable_tasks * sizeof(float)); + if (status != cudaSuccess) + { + std::cerr << "CUDA Runtime Error at: " + << __FILE__ + << ":" + << __LINE__ + << std::endl; + std::cerr << cudaGetErrorString(status) << std::endl; + } + assert((status == cudaSuccess)); + + + //kernel_spike_generation<<>>( + kernel_spike_generation<<>>( + h_init_ptr, + h_pointer, + h_dim_c0, + h_dim_c1, + h_dim_c2, + h_dim, + weights_pointer, + weights_dim_c0, + input_pointer, + input_dim_c0, + input_dim_c1, + input_dim_c2, + epsilon_xy_pointer, + epsilon_xy_dim_c0, + epsilon_xy_dim_c1, + epsilon_t_pointer, + number_of_spikes, + forgetting_offset, + forgetting_offset_local, + dim_x, + dim_y, + (dim_x * dim_y), + //number_of_pattern + max_threadable_tasks, + temp_memory_a, + temp_memory_b + ); + + status = cudaDeviceSynchronize(); + if (status != cudaSuccess) + { + std::cerr << "CUDA Runtime Error at: " + << __FILE__ + << ":" + << __LINE__ + << std::endl; + std::cerr << cudaGetErrorString(status) << std::endl; + } + assert((status == cudaSuccess)); + + status = cudaFree(temp_memory_a); + if (status != cudaSuccess) + { + std::cerr << "CUDA Runtime Error at: " + << __FILE__ + << ":" + << __LINE__ + << std::endl; + std::cerr << cudaGetErrorString(status) << std::endl; + } + assert((status == cudaSuccess)); + + status = cudaFree(temp_memory_b); + if (status != cudaSuccess) + { + std::cerr << "CUDA Runtime Error at: " + << __FILE__ + << ":" + << __LINE__ + << std::endl; + std::cerr << cudaGetErrorString(status) << std::endl; + } + assert((status == cudaSuccess)); + + return; +}; + + +void HDynamicCNNGPU::gpu_occupancy_export( + size_t dim_x, + size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1) +{ + return; +}; + +void HDynamicCNNGPU::gpu_occupancy_import( + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1 +) +{ + return; +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.h b/network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.h new file mode 100644 index 0000000..235bad4 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v1/HDynamicCNNGPU.h @@ -0,0 +1,84 @@ +#ifndef HDYNAMICCNNGPU +#define HDYNAMICCNNGPU + +#include + +#include +#include + +class HDynamicCNNGPU +{ +public: + HDynamicCNNGPU(); + ~HDynamicCNNGPU(); + + void entrypoint( + int64_t h_pointer_addr, + int64_t h_dim_0, + int64_t h_dim_1, + int64_t h_dim_2, + int64_t h_dim_3, + int64_t epsilon_xy_pointer_addr, + int64_t epsilon_xy_dim_0, + int64_t epsilon_xy_dim_1, + int64_t epsilon_xy_dim_2, + int64_t epsilon_t_pointer_addr, + int64_t epsilon_t_dim_0, + int64_t weights_pointer_addr, + int64_t weights_dim_0, + int64_t weights_dim_1, + int64_t input_pointer_addr, + int64_t input_dim_0, + int64_t input_dim_1, + int64_t input_dim_2, + int64_t input_dim_3, + int64_t init_vector_pointer_addr, + int64_t init_vector_dim_0, + int64_t number_of_processes, + float forgetting_offset, + int64_t gpu_tuning_factor); + + void gpu_occupancy_export( + size_t dim_x, + size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1); + + void gpu_occupancy_import( + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1); + +private: + + void gpu_update( + float* h_init_ptr, + float* h_pointer, + size_t h_dim_c0, + size_t h_dim_c1, + size_t h_dim_c2, + size_t h_dim, + float* epsilon_xy_pointer, + size_t epsilon_xy_dim_c0, + size_t epsilon_xy_dim_c1, + float* epsilon_t_pointer, + float* weights_pointer, + size_t weights_dim_c0, + int64_t* input_pointer, + size_t input_dim_c0, + size_t input_dim_c1, + size_t input_dim_c2, + size_t number_of_spikes, + size_t dim_x, + size_t dim_y, + float forgetting_offset, + float forgetting_offset_local, + size_t number_of_pattern, + size_t gpu_tuning_factor); + +}; + +#endif /* HDYNAMICCNNGPU */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v1/Makefile b/network/h_dynamic_cnn_gpu_cpp_v1/Makefile new file mode 100644 index 0000000..acdc799 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v1/Makefile @@ -0,0 +1,33 @@ +include ../.env +export + +name = HDynamicCNN +type = GPU + +PYPOSTFIX := $(shell $(PYBIN)python3-config --extension-suffix) +PYBIND11INCLUDE := $(shell $(PYBIN)python3 -m pybind11 --includes) +PARAMETERS_O = $(PARAMETERS_O_GPU) $(PYBIND11INCLUDE) +PARAMETERS_Linker = $(PARAMETERS_Linker_GPU) + +so_file = Py$(name)$(type)$(PYPOSTFIX) +pyi_file = Py$(name)$(type).pyi +all: ../$(so_file) + +$(O_DIRS)$(name)$(type).o: $(name)$(type).h $(name)$(type).cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c $(name)$(type).cu -o $(O_DIRS)$(name)$(type).o + +$(O_DIRS)Py$(name)$(type).o: $(name)$(type).h Py$(name)$(type).cpp + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c Py$(name)$(type).cpp -o $(O_DIRS)Py$(name)$(type).o + +../$(so_file): $(O_DIRS)$(name)$(type).o $(O_DIRS)Py$(name)$(type).o + $(NVCC) $(PARAMETERS_Linker) -o ../$(so_file) $(O_DIRS)$(name)$(type).o $(O_DIRS)Py$(name)$(type).o + + +####################### +clean: + rm -rf $(O_DIRS) + rm -f ../$(so_file) + rm -f ../$(pyi_file) + diff --git a/network/h_dynamic_cnn_gpu_cpp_v1/PyHDynamicCNNGPU.cpp b/network/h_dynamic_cnn_gpu_cpp_v1/PyHDynamicCNNGPU.cpp new file mode 100644 index 0000000..999b6a6 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v1/PyHDynamicCNNGPU.cpp @@ -0,0 +1,18 @@ +#include + +#include "HDynamicCNNGPU.h" + +namespace py = pybind11; + +PYBIND11_MODULE(PyHDynamicCNNGPU, m) +{ + m.doc() = "HDynamicCNNGPU Module"; + py::class_(m, "HDynamicCNNGPU") + .def(py::init<>()) + .def("gpu_occupancy_export", + &HDynamicCNNGPU::gpu_occupancy_export) + .def("gpu_occupancy_import", + &HDynamicCNNGPU::gpu_occupancy_import) + .def("update", + &HDynamicCNNGPU::entrypoint); +} \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.cu new file mode 100644 index 0000000..c344f1e --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.cu @@ -0,0 +1,727 @@ +#include +#include +#include +#include + +#include +#include +#include + +#include "HDynamicCNNGPU.h" +#include "kernel_phxy_fill_with_h.h" +#include "kernel_phxy_fill_with_spike_selected_w.h" +#include "kernel_phxy_one_over_sum_into_pxy.h" +#include "kernel_phxy_plus_phxy.h" +#include "kernel_phxy_plus_pxy.h" +#include "kernel_phxy_times_phxy_equals_phxy.h" +#include "kernel_phxy_times_pxy.h" +#include "kernel_pxy_plus_v.h" +#include "kernel_pxy_reciprocal.h" +#include "kernel_pxy_set_to_v.h" +#include "kernel_pxy_time_pxy.h" +#include "kernel_pxy_times_spike_selected_sxy.h" +#include "kernel_pxy_times_v.h" + +HDynamicCNNGPU::HDynamicCNNGPU() +{ + +}; + +HDynamicCNNGPU::~HDynamicCNNGPU() +{ + +}; + +void HDynamicCNNGPU::entrypoint( + int64_t h_pointer_addr, + int64_t h_dim_0, + int64_t h_dim_1, + int64_t h_dim_2, + int64_t h_dim_3, + int64_t epsilon_xy_pointer_addr, + int64_t epsilon_xy_dim_0, + int64_t epsilon_xy_dim_1, + int64_t epsilon_xy_dim_2, + int64_t epsilon_t_pointer_addr, + int64_t epsilon_t_dim_0, + int64_t weights_pointer_addr, + int64_t weights_dim_0, + int64_t weights_dim_1, + int64_t input_pointer_addr, + int64_t input_dim_0, + int64_t input_dim_1, + int64_t input_dim_2, + int64_t input_dim_3, + int64_t init_vector_pointer_addr, + int64_t init_vector_dim_0, + int64_t number_of_processes, + float forgetting_offset, + int64_t gpu_tuning_factor +) +{ + std::cout << "Hello\n"; + + size_t number_of_pattern = input_dim_0; + + size_t h_dim = init_vector_dim_0; + float* h_init_ptr = (float*)init_vector_pointer_addr; + assert((h_init_ptr != nullptr)); + assert((h_dim > 0)); + + float* h_pointer = (float*)h_pointer_addr; + assert((h_pointer != nullptr)); + assert((h_dim_0 > 0)); + assert((h_dim_1 > 0)); + assert((h_dim_2 > 0)); + assert((h_dim_3 > 0)); + + size_t h_dim_c0 = h_dim_1 * h_dim_2 * h_dim_3; + size_t h_dim_c1 = h_dim_2 * h_dim_3; + size_t h_dim_c2 = h_dim_3; + + float* epsilon_xy_pointer = nullptr; + size_t epsilon_xy_dim_c0 = 0; + size_t epsilon_xy_dim_c1 = 0; + if (epsilon_xy_pointer_addr != 0) + { + epsilon_xy_pointer = (float*)epsilon_xy_pointer_addr; + assert((epsilon_xy_pointer != nullptr)); + assert((epsilon_xy_dim_0 > 0)); + assert((epsilon_xy_dim_1 > 0)); + assert((epsilon_xy_dim_2 > 0)); + + epsilon_xy_dim_c0 = epsilon_xy_dim_2 * epsilon_xy_dim_1; + epsilon_xy_dim_c1 = epsilon_xy_dim_2; + } + + float* epsilon_t_pointer = (float*)epsilon_t_pointer_addr; + assert((epsilon_t_pointer != nullptr)); + assert((epsilon_t_dim_0 > 0)); + + float* weights_pointer = (float*)weights_pointer_addr; + assert((weights_pointer != nullptr)); + assert((weights_dim_0 > 0)); + assert((weights_dim_1 > 0)); + + size_t weights_dim_c0 = weights_dim_1; + + int64_t* input_pointer = (int64_t*)input_pointer_addr; + assert((input_pointer != nullptr)); + assert((input_dim_0 > 0)); + assert((input_dim_1 > 0)); + assert((input_dim_2 > 0)); + assert((input_dim_3 > 0)); + + size_t input_dim_c0 = input_dim_1 * input_dim_2 * input_dim_3; + size_t input_dim_c1 = input_dim_2 * input_dim_3; + size_t input_dim_c2 = input_dim_3; + + assert((h_dim == weights_dim_1)); + size_t number_of_spikes = input_dim_1; + size_t dim_x = input_dim_2; + size_t dim_y = input_dim_3; + + float forgetting_offset_local = forgetting_offset / static_cast(h_dim); + + // -------------------- + assert((number_of_processes <= 0)); + gpu_update(h_init_ptr, h_pointer, h_dim_c0, h_dim_c1, h_dim_c2, h_dim, + epsilon_xy_pointer, epsilon_xy_dim_c0, epsilon_xy_dim_c1, + epsilon_t_pointer, weights_pointer, weights_dim_c0, + input_pointer, input_dim_c0, input_dim_c1, input_dim_c2, + number_of_spikes, dim_x, dim_y, forgetting_offset, + forgetting_offset_local, number_of_pattern, gpu_tuning_factor); + return; +}; + + +void HDynamicCNNGPU::gpu_occupancy_measure( + size_t dim_x, + size_t dim_y, + size_t number_of_pattern, + size_t h_dim) +{ + grid_and_thread_calculated = false; + assert((dim_x < 65535)); + assert((dim_y < 65535)); + + grid_and_thread_settings.resize(14); + + occupancy_kernel_phxy_plus_phxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY], display_debug); + + occupancy_kernel_pxy_plus_v(dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V], + display_debug); + + occupancy_kernel_pxy_times_v(dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V], + display_debug); + + occupancy_kernel_phxy_fill_with_h( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H], display_debug); + + occupancy_kernel_phxy_plus_pxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY], display_debug); + + occupancy_kernel_pxy_reciprocal( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL], display_debug); + + occupancy_kernel_phxy_fill_with_spike_selected_w( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W], + display_debug); + + occupancy_kernel_phxy_times_phxy_equals_phxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY], + display_debug); + + occupancy_kernel_pxy_set_to_v( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V], display_debug); + + occupancy_kernel_phxy_one_over_sum_into_pxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY], + display_debug); + + occupancy_kernel_phxy_times_pxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY], display_debug); + + occupancy_kernel_pxy_time_pxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY], display_debug); + + // occupancy_kernel_approximation_pure_multiplication( + // dim_x, dim_y, number_of_pattern, h_dim, + // grid_and_thread_settings[ID_KERNEL_APPROXIMATION_MULTIPLICATION], + // display_debug); + + occupancy_kernel_pxy_times_spike_selected_sxy( + dim_x, dim_y, number_of_pattern, h_dim, + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY], + display_debug); + + grid_and_thread_calculated = true; + return; +}; + +void HDynamicCNNGPU::gpu_occupancy_export( + size_t dim_x, + size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1) +{ + int64_t* setting_memory = (int64_t*)setting_memory_addr; + + assert((setting_memory != nullptr)); + assert((setting_dim_1 == H_DYNAMIC_NUMBER_OF_KERNELS_PARAMETERS)); + + gpu_occupancy_measure(dim_x, dim_y, number_of_pattern, h_dim); + assert((grid_and_thread_calculated == true)); + + assert((setting_dim_0 == grid_and_thread_settings.size())); + + for (size_t counter_0 = 0; counter_0 < setting_dim_0; counter_0++) + { + for (size_t counter_1 = 0; counter_1 < setting_dim_1; counter_1++) + { + setting_memory[counter_0 * setting_dim_1 + counter_1] = + grid_and_thread_settings[counter_0][counter_1]; + } + } +}; + +void HDynamicCNNGPU::gpu_occupancy_import( + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1) +{ + grid_and_thread_calculated = false; + + int64_t* setting_memory = (int64_t*)setting_memory_addr; + + assert((setting_memory != nullptr)); + assert((setting_dim_1 == H_DYNAMIC_NUMBER_OF_KERNELS_PARAMETERS)); + assert((setting_dim_0 == H_DYNAMIC_NUMBER_OF_KERNELS)); + + grid_and_thread_settings.resize(H_DYNAMIC_NUMBER_OF_KERNELS); + + for (size_t counter_0 = 0; counter_0 < setting_dim_0; counter_0++) + { + grid_and_thread_settings[counter_0].resize( + H_DYNAMIC_NUMBER_OF_KERNELS_PARAMETERS); + + for (size_t counter_1 = 0; counter_1 < setting_dim_1; counter_1++) + { + grid_and_thread_settings[counter_0][counter_1] = + setting_memory[counter_0 * setting_dim_1 + counter_1]; + } + } + + grid_and_thread_calculated = true; +}; + +void HDynamicCNNGPU::gpu_update( + float* h_init_ptr, + float* h_pointer, + size_t h_dim_c0, + size_t h_dim_c1, + size_t h_dim_c2, + size_t h_dim, + float* epsilon_xy_pointer, + size_t epsilon_xy_dim_c0, + size_t epsilon_xy_dim_c1, + float* epsilon_t_pointer, + float* weights_pointer, + size_t weights_dim_c0, + int64_t* input_pointer, + size_t input_dim_c0, + size_t input_dim_c1, + size_t input_dim_c2, + size_t number_of_spikes, + size_t dim_x, + size_t dim_y, + float forgetting_offset, + float forgetting_offset_local, + size_t number_of_pattern, + size_t gpu_tuning_factor) +{ + std::cout << "0\n"; + if (grid_and_thread_calculated == false) + { + gpu_occupancy_measure(dim_x, dim_y, number_of_pattern, h_dim); + } + assert((grid_and_thread_calculated == true)); + + cudaError_t status; + + size_t h_sum_dim_c0 = dim_x * dim_y; + size_t h_sum_dim_c1 = dim_y; + + size_t phxy_block_dim_c0 = h_dim * dim_x * dim_y; + size_t phxy_block_dim_c1 = dim_x * dim_y; + size_t phxy_block_dim_c2 = dim_y; + + size_t pxy_block_dim_c0 = dim_x * dim_y; + size_t pxy_block_dim_c1 = dim_y; + + std::cout << "1\n"; + float* w_memory = nullptr; + status = cudaMalloc((void**)&w_memory, number_of_pattern * h_dim * dim_x * + dim_y * sizeof(float)); + assert((status == cudaSuccess)); + std::cout << "2\n"; + float* h_temp_memory = nullptr; + status = + cudaMalloc((void**)&h_temp_memory, + number_of_pattern * h_dim * dim_x * dim_y * sizeof(float)); + assert((status == cudaSuccess)); + + std::cout << "3\n"; + float* h_sum_memory = nullptr; + status = cudaMalloc((void**)&h_sum_memory, + number_of_pattern * dim_x * dim_y * sizeof(float)); + assert((status == cudaSuccess)); + + std::cout << "4\n"; + float* epsilon_subsegment_memory = nullptr; + status = cudaMalloc((void**)&epsilon_subsegment_memory, + number_of_pattern * dim_x * dim_y * sizeof(float)); + assert((status == cudaSuccess)); + + std::cout << "5\n"; + float* epsilon_scale_memory = nullptr; + status = cudaMalloc((void**)&epsilon_scale_memory, + number_of_pattern * dim_x * dim_y * sizeof(float)); + assert((status == cudaSuccess)); + + + std::cout << "6\n"; + float* forget_memory = nullptr; + if (forgetting_offset > 0.0) + { + status = cudaMalloc((void**)&forget_memory, + number_of_pattern * dim_x * dim_y * sizeof(float)); + assert((status == cudaSuccess)); + } + // --- + std::cout << "A\n"; + // Initialize h + kernel_phxy_fill_with_h<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][0], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][1], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][3], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][4], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][5])>>>( + h_init_ptr, h_pointer, h_dim_c0, h_dim_c1, h_dim_c2, h_dim, + phxy_block_dim_c0, phxy_block_dim_c1, phxy_block_dim_c2, + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_H][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "B\n"; + // Set epsilon memory scale to 1.0 + kernel_pxy_set_to_v<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][0], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][1], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][3], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][4], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][5])>>>( + epsilon_scale_memory, 1.0, + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "C\n"; + for (size_t counter_spike = 0; counter_spike < number_of_spikes; + counter_spike++) + { + // Get epsilon_t from gpu memory + float epsilon_t; + status = cudaMemcpy(&epsilon_t, &epsilon_t_pointer[counter_spike], + sizeof(float), cudaMemcpyDeviceToHost); + assert((status == cudaSuccess)); + // Set epsilon memory subsegment to epsilon(t) + kernel_pxy_set_to_v<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][0], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][1], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][3], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][4], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][5])>>>( + epsilon_subsegment_memory, epsilon_t, + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "D\n"; + if (forget_memory != nullptr) + { + // Set forget memory subsegment to forgetting_offset_local + kernel_pxy_set_to_v<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][0], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][1], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][3], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][4], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][5])>>>( + forget_memory, forgetting_offset_local, + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + } + + std::cout << "E\n"; + // if (*spike >= 0) { + // epsilon_subsegment = *epsilon_xy_pointer[*spike * + // epsilon_xy_dim_c0] + if (epsilon_xy_dim_c0 != 0) + { + kernel_pxy_times_spike_selected_sxy<<< + dim3( + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY] + [2]), + dim3( + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY] + [5])>>>( + epsilon_subsegment_memory, epsilon_xy_pointer, input_pointer, + counter_spike, input_dim_c0, input_dim_c1, input_dim_c2, + epsilon_xy_dim_c0, epsilon_xy_dim_c1, epsilon_xy_dim_c0, + epsilon_xy_dim_c1, pxy_block_dim_c0, pxy_block_dim_c1, + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + } + std::cout << "F\n"; + // Get the weight vectors according the spikes + kernel_phxy_fill_with_spike_selected_w<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W] + [0], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W] + [1], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W] + [2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W] + [3], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W] + [4], + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W] + [5])>>>( + w_memory, weights_pointer, input_pointer, counter_spike, weights_dim_c0, + input_dim_c0, input_dim_c1, input_dim_c2, h_dim_c0, h_dim_c1, h_dim_c2, + h_dim, phxy_block_dim_c0, phxy_block_dim_c1, phxy_block_dim_c2, + grid_and_thread_settings[ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "G\n"; + // h_temp = h * w + kernel_phxy_times_phxy_equals_phxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY] + [0], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY] + [1], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY] + [2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY] + [3], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY] + [4], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY] + [5])>>>( + h_pointer, w_memory, h_temp_memory, + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "H\n"; + // 1 / sum h_temp + kernel_phxy_one_over_sum_into_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY][0], + grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY][1], + grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY][3], + grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY][4], + grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY] + [5])>>>( + h_temp_memory, h_sum_memory, h_dim_c0, h_dim_c1, h_dim_c2, h_dim, + h_sum_dim_c0, h_sum_dim_c1, pxy_block_dim_c0, pxy_block_dim_c1, + grid_and_thread_settings[ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "I\n"; + // epsilon_scale / sum h_temp + kernel_pxy_time_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][5])>>>( + h_sum_memory, epsilon_scale_memory, + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "J\n"; + // epsilon_subsegment * epsilon_scale / sum h_temp + kernel_pxy_time_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][5])>>>( + h_sum_memory, epsilon_subsegment_memory, + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "K\n"; + // epsilon_scale * forget_memory which contains forgetting_offset_local + if (forget_memory != nullptr) + { + kernel_pxy_time_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][5])>>>( + forget_memory, epsilon_scale_memory, + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + } + + std::cout << "L\n"; + // delta_forget = epsilon_subsegment * epsilon_scale * forget_memory + if (forget_memory != nullptr) + { + kernel_pxy_time_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][5])>>>( + forget_memory, epsilon_subsegment_memory, + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + } + std::cout << "M\n"; + // delta_h = h_temp_memory * epsilon_subsegment * epsilon_scale / sum h + kernel_phxy_times_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][0], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][1], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][3], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][4], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][5])>>>( + h_temp_memory, h_sum_memory, h_dim_c0, h_dim_c1, h_dim_c2, h_dim, + h_sum_dim_c0, h_sum_dim_c1, phxy_block_dim_c0, phxy_block_dim_c1, + phxy_block_dim_c2, + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "N\n"; + // h + delta_h + kernel_phxy_plus_phxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][0], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][1], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][3], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][4], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][5])>>>( + h_pointer, h_temp_memory, + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PHXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "O\n"; + // h + delta_h + delta_forget + kernel_phxy_plus_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][0], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][1], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][3], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][4], + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][5])>>>( + h_pointer, forget_memory, h_dim_c0, h_dim_c1, h_dim_c2, h_dim, + h_sum_dim_c0, h_sum_dim_c1, phxy_block_dim_c0, phxy_block_dim_c1, + phxy_block_dim_c2, + grid_and_thread_settings[ID_KERNEL_PHXY_PLUS_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "P\n"; + kernel_pxy_times_v<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][5])>>>( + epsilon_subsegment_memory, (1.0 + forgetting_offset), + grid_and_thread_settings[ID_KERNEL_PXY_TIMES_V][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "Q\n"; + kernel_pxy_plus_v<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][0], + grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][1], + grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][3], + grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][4], + grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][5])>>>( + epsilon_subsegment_memory, 1.0, + grid_and_thread_settings[ID_KERNEL_PXY_PLUS_V][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "R\n"; + // epsilon_scale * epsilon_subsegment + kernel_pxy_time_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][0], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][1], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][3], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][4], + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][5])>>>( + epsilon_scale_memory, epsilon_subsegment_memory, + grid_and_thread_settings[ID_KERNEL_PXY_TIME_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + if (((counter_spike > 0) && (counter_spike % 5000 == 0)) || + (counter_spike + 1 == number_of_spikes)) + { + std::cout << "S\n"; + kernel_pxy_reciprocal<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][0], + grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][1], + grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][3], + grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][4], + grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][5])>>>( + epsilon_scale_memory, + grid_and_thread_settings[ID_KERNEL_PXY_RECIPROCAL][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "T\n"; + kernel_phxy_times_pxy<<< + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][0], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][1], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][3], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][4], + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][5])>>>( + h_pointer, epsilon_scale_memory, h_dim_c0, h_dim_c1, h_dim_c2, h_dim, + h_sum_dim_c0, h_sum_dim_c1, phxy_block_dim_c0, phxy_block_dim_c1, + phxy_block_dim_c2, + grid_and_thread_settings[ID_KERNEL_PHXY_TIMES_PXY][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + + std::cout << "U\n"; + // Set epsilon memory scale to 1.0 + kernel_pxy_set_to_v<<< + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][0], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][1], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][2]), + dim3(grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][3], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][4], + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][5])>>>( + epsilon_scale_memory, 1.0, + grid_and_thread_settings[ID_KERNEL_PXY_SET_TO_V][6]); + status = cudaDeviceSynchronize(); + assert((status == cudaSuccess)); + } + } + std::cout << "V\n"; + // ------------ + + status = cudaFree(w_memory); + assert((status == cudaSuccess)); + + status = cudaFree(h_temp_memory); + assert((status == cudaSuccess)); + + status = cudaFree(h_sum_memory); + assert((status == cudaSuccess)); + + status = cudaFree(epsilon_subsegment_memory); + assert((status == cudaSuccess)); + + status = cudaFree(epsilon_scale_memory); + assert((status == cudaSuccess)); + + if (forget_memory != nullptr) + { + status = cudaFree(forget_memory); + assert((status == cudaSuccess)); + } + + return; +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.h new file mode 100644 index 0000000..2c758ea --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/HDynamicCNNGPU.h @@ -0,0 +1,113 @@ +#ifndef HDYNAMICCNNGPU +#define HDYNAMICCNNGPU + +#include +#include + +#include +#include +#include + +#define ID_KERNEL_PHXY_PLUS_PHXY 0 +#define ID_KERNEL_PXY_PLUS_V 1 +#define ID_KERNEL_PXY_TIMES_V 2 +#define ID_KERNEL_PHXY_FILL_WITH_H 3 +#define ID_KERNEL_PHXY_PLUS_PXY 4 +#define ID_KERNEL_PXY_RECIPROCAL 5 +#define ID_KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W 6 +#define ID_KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY 7 +#define ID_KERNEL_PXY_SET_TO_V 8 +#define ID_KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY 9 +#define ID_KERNEL_PHXY_TIMES_PXY 10 +#define ID_KERNEL_PXY_TIME_PXY 11 +#define ID_KERNEL_APPROXIMATION_MULTIPLICATION 12 +#define ID_KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY 13 + +#define H_DYNAMIC_NUMBER_OF_KERNELS 14 +#define H_DYNAMIC_NUMBER_OF_KERNELS_PARAMETERS 7 + +class HDynamicCNNGPU +{ + public: + HDynamicCNNGPU(); + ~HDynamicCNNGPU(); + + void entrypoint( + int64_t h_pointer_addr, + int64_t h_dim_0, + int64_t h_dim_1, + int64_t h_dim_2, + int64_t h_dim_3, + int64_t epsilon_xy_pointer_addr, + int64_t epsilon_xy_dim_0, + int64_t epsilon_xy_dim_1, + int64_t epsilon_xy_dim_2, + int64_t epsilon_t_pointer_addr, + int64_t epsilon_t_dim_0, + int64_t weights_pointer_addr, + int64_t weights_dim_0, + int64_t weights_dim_1, + int64_t input_pointer_addr, + int64_t input_dim_0, + int64_t input_dim_1, + int64_t input_dim_2, + int64_t input_dim_3, + int64_t init_vector_pointer_addr, + int64_t init_vector_dim_0, + int64_t number_of_processes, + float forgetting_offset, + int64_t gpu_tuning_factor + ); + + void gpu_occupancy_export( + size_t dim_x, + size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1); + + void gpu_occupancy_import( + int64_t setting_memory_addr, + size_t setting_dim_0, + size_t setting_dim_1); + + private: + + void gpu_update( + float* h_init_ptr, + float* h_pointer, + size_t h_dim_c0, + size_t h_dim_c1, + size_t h_dim_c2, + size_t h_dim, + float* epsilon_xy_pointer, + size_t epsilon_xy_dim_c0, + size_t epsilon_xy_dim_c1, + float* epsilon_t_pointer, + float* weights_pointer, + size_t weights_dim_c0, + int64_t* input_pointer, + size_t input_dim_c0, + size_t input_dim_c1, + size_t input_dim_c2, + size_t number_of_spikes, + size_t dim_x, size_t dim_y, + float forgetting_offset, + float forgetting_offset_local, + size_t number_of_pattern, + size_t gpu_tuning_factor); + + void gpu_occupancy_measure( + size_t dim_x, + size_t dim_y, + size_t number_of_pattern, + size_t h_dim); + + bool grid_and_thread_calculated = false; + std::vector> grid_and_thread_settings; + bool display_debug = false; +}; + +#endif /* HDYNAMICCNNGPU */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/Makefile b/network/h_dynamic_cnn_gpu_cpp_v2_pre/Makefile new file mode 100644 index 0000000..11e1771 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/Makefile @@ -0,0 +1,153 @@ +include ../.env +export + +name = HDynamicCNN +type = GPU + +PYPOSTFIX := $(shell $(PYBIN)python3-config --extension-suffix) +PYBIND11INCLUDE := $(shell $(PYBIN)python3 -m pybind11 --includes) +PARAMETERS_O = $(PARAMETERS_O_GPU) $(PYBIND11INCLUDE) +PARAMETERS_Linker = $(PARAMETERS_Linker_GPU) + +so_file = Py$(name)$(type)$(PYPOSTFIX) +pyi_file = Py$(name)$(type).pyi +all: ../$(so_file) + + +$(O_DIRS)$(name)$(type).o: \ + $(name)$(type).h \ + $(name)$(type).cu \ + kernel_helper_functions.h \ + kernel_phxy_plus_pxy.h \ + kernel_pxy_set_to_v.h \ + kernel_phxy_fill_with_h.h \ + kernel_phxy_times_phxy_equals_phxy.h \ + kernel_pxy_time_pxy.h \ + kernel_phxy_fill_with_spike_selected_w.h \ + kernel_phxy_times_pxy.h \ + kernel_pxy_times_spike_selected_sxy.h \ + kernel_phxy_one_over_sum_into_pxy.h \ + kernel_pxy_plus_v.h \ + kernel_pxy_times_v.h \ + kernel_phxy_plus_phxy.h \ + kernel_pxy_reciprocal.h + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c $(name)$(type).cu -o $(O_DIRS)$(name)$(type).o + +$(O_DIRS)Py$(name)$(type).o: $(name)$(type).h Py$(name)$(type).cpp + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c Py$(name)$(type).cpp -o $(O_DIRS)Py$(name)$(type).o + +../$(so_file): \ + $(O_DIRS)$(name)$(type).o \ + $(O_DIRS)Py$(name)$(type).o \ + $(O_DIRS)kernel_helper_functions.o \ + $(O_DIRS)kernel_phxy_plus_pxy.o \ + $(O_DIRS)kernel_pxy_set_to_v.o \ + $(O_DIRS)kernel_phxy_fill_with_h.o \ + $(O_DIRS)kernel_phxy_times_phxy_equals_phxy.o \ + $(O_DIRS)kernel_pxy_time_pxy.o \ + $(O_DIRS)kernel_phxy_fill_with_spike_selected_w.o \ + $(O_DIRS)kernel_phxy_times_pxy.o \ + $(O_DIRS)kernel_pxy_times_spike_selected_sxy.o \ + $(O_DIRS)kernel_phxy_one_over_sum_into_pxy.o \ + $(O_DIRS)kernel_pxy_plus_v.o \ + $(O_DIRS)kernel_pxy_times_v.o \ + $(O_DIRS)kernel_phxy_plus_phxy.o \ + $(O_DIRS)kernel_pxy_reciprocal.o + $(NVCC) $(PARAMETERS_Linker) -o ../$(so_file) \ + $(O_DIRS)$(name)$(type).o \ + $(O_DIRS)Py$(name)$(type).o \ + $(O_DIRS)kernel_helper_functions.o \ + $(O_DIRS)kernel_phxy_plus_pxy.o \ + $(O_DIRS)kernel_pxy_set_to_v.o \ + $(O_DIRS)kernel_phxy_fill_with_h.o \ + $(O_DIRS)kernel_phxy_times_phxy_equals_phxy.o \ + $(O_DIRS)kernel_pxy_time_pxy.o \ + $(O_DIRS)kernel_phxy_fill_with_spike_selected_w.o \ + $(O_DIRS)kernel_phxy_times_pxy.o \ + $(O_DIRS)kernel_pxy_times_spike_selected_sxy.o \ + $(O_DIRS)kernel_phxy_one_over_sum_into_pxy.o \ + $(O_DIRS)kernel_pxy_plus_v.o \ + $(O_DIRS)kernel_pxy_times_v.o \ + $(O_DIRS)kernel_phxy_plus_phxy.o \ + $(O_DIRS)kernel_pxy_reciprocal.o + + +$(O_DIRS)kernel_helper_functions.o: kernel_helper_functions.h kernel_helper_functions.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_helper_functions.cu -o $(O_DIRS)kernel_helper_functions.o + +$(O_DIRS)kernel_phxy_plus_pxy.o: kernel_helper_functions.h \ + kernel_phxy_plus_pxy.h kernel_phxy_plus_pxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_plus_pxy.cu -o $(O_DIRS)kernel_phxy_plus_pxy.o + +$(O_DIRS)kernel_pxy_set_to_v.o: kernel_helper_functions.h \ + kernel_pxy_set_to_v.h kernel_pxy_set_to_v.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_pxy_set_to_v.cu -o $(O_DIRS)kernel_pxy_set_to_v.o + +$(O_DIRS)kernel_phxy_fill_with_h.o: kernel_helper_functions.h \ + kernel_phxy_fill_with_h.h kernel_phxy_fill_with_h.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_fill_with_h.cu -o $(O_DIRS)kernel_phxy_fill_with_h.o + +$(O_DIRS)kernel_phxy_times_phxy_equals_phxy.o: kernel_helper_functions.h \ + kernel_phxy_times_phxy_equals_phxy.h kernel_phxy_times_phxy_equals_phxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_times_phxy_equals_phxy.cu -o $(O_DIRS)kernel_phxy_times_phxy_equals_phxy.o + +$(O_DIRS)kernel_pxy_time_pxy.o: kernel_helper_functions.h \ + kernel_pxy_time_pxy.h kernel_pxy_time_pxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_pxy_time_pxy.cu -o $(O_DIRS)kernel_pxy_time_pxy.o + +$(O_DIRS)kernel_phxy_fill_with_spike_selected_w.o: kernel_helper_functions.h \ + kernel_phxy_fill_with_spike_selected_w.h kernel_phxy_fill_with_spike_selected_w.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_fill_with_spike_selected_w.cu -o $(O_DIRS)kernel_phxy_fill_with_spike_selected_w.o + +$(O_DIRS)kernel_phxy_times_pxy.o: kernel_helper_functions.h \ + kernel_phxy_times_pxy.h kernel_phxy_times_pxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_times_pxy.cu -o $(O_DIRS)kernel_phxy_times_pxy.o + +$(O_DIRS)kernel_pxy_times_spike_selected_sxy.o: kernel_helper_functions.h \ + kernel_pxy_times_spike_selected_sxy.h kernel_pxy_times_spike_selected_sxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_pxy_times_spike_selected_sxy.cu -o $(O_DIRS)kernel_pxy_times_spike_selected_sxy.o + +$(O_DIRS)kernel_phxy_one_over_sum_into_pxy.o: kernel_helper_functions.h \ + kernel_phxy_one_over_sum_into_pxy.h kernel_phxy_one_over_sum_into_pxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_one_over_sum_into_pxy.cu -o $(O_DIRS)kernel_phxy_one_over_sum_into_pxy.o + +$(O_DIRS)kernel_pxy_plus_v.o: kernel_helper_functions.h \ + kernel_pxy_plus_v.h kernel_pxy_plus_v.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_pxy_plus_v.cu -o $(O_DIRS)kernel_pxy_plus_v.o + +$(O_DIRS)kernel_pxy_times_v.o: kernel_helper_functions.h \ + kernel_pxy_times_v.h kernel_pxy_times_v.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_pxy_times_v.cu -o $(O_DIRS)kernel_pxy_times_v.o + +$(O_DIRS)kernel_phxy_plus_phxy.o: kernel_helper_functions.h \ + kernel_phxy_plus_phxy.h kernel_phxy_plus_phxy.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_phxy_plus_phxy.cu -o $(O_DIRS)kernel_phxy_plus_phxy.o + +$(O_DIRS)kernel_pxy_reciprocal.o: kernel_helper_functions.h \ + kernel_pxy_reciprocal.h kernel_pxy_reciprocal.cu + mkdir -p $(O_DIRS) + $(NVCC) $(PARAMETERS_O) -c kernel_pxy_reciprocal.cu -o $(O_DIRS)kernel_pxy_reciprocal.o + + + +####################### +clean: + rm -rf $(O_DIRS) + rm -f ../$(so_file) + rm -f ../$(pyi_file) + diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/PyHDynamicCNNGPU.cpp b/network/h_dynamic_cnn_gpu_cpp_v2_pre/PyHDynamicCNNGPU.cpp new file mode 100644 index 0000000..d6676a2 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/PyHDynamicCNNGPU.cpp @@ -0,0 +1,18 @@ +#include + +#include "HDynamicCNNGPU.h" + +namespace py = pybind11; + +PYBIND11_MODULE(PyHDynamicCNNGPU, m) +{ + m.doc() = "HDynamicCNNManyIP Module"; + py::class_(m, "HDynamicCNNGPU") + .def(py::init<>()) + .def("gpu_occupancy_export", + &HDynamicCNNGPU::gpu_occupancy_export) + .def("gpu_occupancy_import", + &HDynamicCNNGPU::gpu_occupancy_import) + .def("update", + &HDynamicCNNGPU::entrypoint); +} \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.cu new file mode 100644 index 0000000..bd023e9 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.cu @@ -0,0 +1,17 @@ +#include + +#include "kernel_helper_functions.h" + +void kernel_debug_plot(std::vector output, bool display_debug) { + if (display_debug == true) { + std::cout << "grid x: " << output[0] << std::endl; + std::cout << "grid y: " << output[1] << std::endl; + std::cout << "grid z: " << output[2] << std::endl; + std::cout << "thread block x: " << output[3] << std::endl; + std::cout << "thread block y: " << output[4] << std::endl; + std::cout << "thread block z: " << output[5] << std::endl; + std::cout << "max_idx: " << output[6] << std::endl << std::endl; + } + + return; +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.h new file mode 100644 index 0000000..819910e --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_helper_functions.h @@ -0,0 +1,7 @@ +#ifndef KERNEL_HELPER_FUNCTIONS +#define KERNEL_HELPER_FUNCTIONS +#include + +void kernel_debug_plot(std::vector output, bool display_debug); + +#endif /* KERNEL_HELPER_FUNCTIONS */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.cu new file mode 100644 index 0000000..d2adf37 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.cu @@ -0,0 +1,64 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_fill_with_h.h" + +__global__ void kernel_phxy_fill_with_h(float* __restrict__ h_memory, + float* __restrict__ phxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, + size_t block_dim_c0, + size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + size_t pattern_id = idx / block_dim_c0; + idx -= pattern_id * block_dim_c0; + size_t idx_h = idx / block_dim_c1; + idx -= idx_h * block_dim_c1; + size_t position_x = idx / block_dim_c2; + idx -= position_x * block_dim_c2; + size_t position_y = idx; + + phxy_memory[pattern_id * phxy_dim_c0 + idx_h * phxy_dim_c1 + + position_x * phxy_dim_c2 + position_y] = h_memory[idx_h]; + } +}; + +void occupancy_kernel_phxy_fill_with_h(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * h_dim * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_phxy_fill_with_h, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_fill_with_h:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.h new file mode 100644 index 0000000..cbe8f5f --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_h.h @@ -0,0 +1,18 @@ +#ifndef KERNEL_PHXY_FILL_WITH_H +#define KERNEL_PHXY_FILL_WITH_H +#include + +__global__ void kernel_phxy_fill_with_h(float* __restrict__ h_memory, + float* __restrict__ phxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, + size_t block_dim_c0, + size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx); + +void occupancy_kernel_phxy_fill_with_h(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PHXY_FILL_WITH_H */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.cu new file mode 100644 index 0000000..0f721e1 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.cu @@ -0,0 +1,73 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_fill_with_spike_selected_w.h" + +__global__ void kernel_phxy_fill_with_spike_selected_w( + float* __restrict__ phxy_memory, float* __restrict__ weights_memory, + int64_t* __restrict__ spike_memory, size_t spike_time, + size_t weights_dim_c0, size_t spike_dim_c0, size_t spike_dim_c1, + size_t spike_dim_c2, size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, size_t block_dim_c0, size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + size_t pattern_id = idx / block_dim_c0; + idx -= pattern_id * block_dim_c0; + size_t idx_h = idx / block_dim_c1; + idx -= idx_h * block_dim_c1; + size_t position_x = idx / block_dim_c2; + idx -= position_x * block_dim_c2; + size_t position_y = idx; + + int64_t* spike = spike_memory + pattern_id * spike_dim_c0 + + spike_time * spike_dim_c1 + position_x * spike_dim_c2 + + position_y; + + if (*spike >= 0) { + phxy_memory[pattern_id * phxy_dim_c0 + idx_h * phxy_dim_c1 + + position_x * phxy_dim_c2 + position_y] = + weights_memory[*spike * weights_dim_c0 + idx_h]; + } else { + phxy_memory[pattern_id * phxy_dim_c0 + idx_h * phxy_dim_c1 + + position_x * phxy_dim_c2 + position_y] = 0.0; + } + } +}; + +void occupancy_kernel_phxy_fill_with_spike_selected_w( + size_t dim_x, size_t dim_y, size_t number_of_pattern, size_t h_dim, + std::vector& output, bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * h_dim * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, + (void*)kernel_phxy_fill_with_spike_selected_w, 0, max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_fill_with_spike_selected_w:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.h new file mode 100644 index 0000000..3055238 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_fill_with_spike_selected_w.h @@ -0,0 +1,17 @@ +#ifndef KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W +#define KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W +#include + +__global__ void kernel_phxy_fill_with_spike_selected_w( + float* __restrict__ phxy_memory, float* __restrict__ weights_memory, + int64_t* __restrict__ spike_memory, size_t spike_time, + size_t weights_dim_c0, size_t spike_dim_c0, size_t spike_dim_c1, + size_t spike_dim_c2, size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, size_t block_dim_c0, size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx); + +void occupancy_kernel_phxy_fill_with_spike_selected_w( + size_t dim_x, size_t dim_y, size_t number_of_pattern, size_t h_dim, + std::vector& output, bool display_debug); + +#endif /* KERNEL_PHXY_FILL_WITH_SPIKE_SELECTED_W */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.cu new file mode 100644 index 0000000..475793f --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.cu @@ -0,0 +1,74 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_one_over_sum_into_pxy.h" + +__global__ void kernel_phxy_one_over_sum_into_pxy( + float* __restrict__ phxy_memory, float* __restrict__ pxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, size_t phxy_dim_c2, size_t h_dim, + size_t pxy_dim_c0, size_t pxy_dim_c1, size_t block_dim_c0, + size_t block_dim_c1, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + size_t pattern_id = idx / block_dim_c0; + idx -= pattern_id * block_dim_c0; + size_t position_x = idx / block_dim_c1; + idx -= position_x * block_dim_c1; + size_t position_y = idx; + + size_t offset_phxy_temp = + pattern_id * phxy_dim_c0 + position_x * phxy_dim_c2 + position_y; + + size_t offset_pxy_sum = + pattern_id * pxy_dim_c0 + position_x * pxy_dim_c1 + position_y; + + float temp = 0.0; + for (size_t idx_h = 0; idx_h < h_dim; idx_h++) { + temp += phxy_memory[offset_phxy_temp + idx_h * phxy_dim_c1]; + } + if (temp > 1E-10) { + pxy_memory[offset_pxy_sum] = 1.0 / temp; + } else { + pxy_memory[offset_pxy_sum] = 0.0; + } + } +}; + +void occupancy_kernel_phxy_one_over_sum_into_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, + (void*)kernel_phxy_one_over_sum_into_pxy, 0, max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_one_over_sum_into_pxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.h new file mode 100644 index 0000000..6f91445 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_one_over_sum_into_pxy.h @@ -0,0 +1,17 @@ +#ifndef KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY +#define KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY +#include + +__global__ void kernel_phxy_one_over_sum_into_pxy( + float* __restrict__ phxy_memory, float* __restrict__ pxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, size_t phxy_dim_c2, size_t h_dim, + size_t pxy_dim_c0, size_t pxy_dim_c1, size_t block_dim_c0, + size_t block_dim_c1, size_t max_idx); + +void occupancy_kernel_phxy_one_over_sum_into_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PHXY_ONE_OVER_SUM_INTO_PXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.cu new file mode 100644 index 0000000..f9ee9c2 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.cu @@ -0,0 +1,51 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_plus_phxy.h" + +__global__ void kernel_phxy_plus_phxy(float* __restrict__ phxy_memory_a, + float* __restrict__ phxy_memory_b, + size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + phxy_memory_a[idx] += phxy_memory_b[idx]; + } +}; + +void occupancy_kernel_phxy_plus_phxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * h_dim * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_phxy_plus_phxy, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_plus_phxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.h new file mode 100644 index 0000000..a247c90 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_phxy.h @@ -0,0 +1,12 @@ +#ifndef KERNEL_PHXY_PLUS_PHXY +#define KERNEL_PHXY_PLUS_PHXY +#include +__global__ void kernel_phxy_plus_phxy(float* __restrict__ phxy_memory_a, + float* __restrict__ phxy_memory_b, + size_t max_idx); + +void occupancy_kernel_phxy_plus_phxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); +#endif /* KERNEL_PHXY_PLUS_PHXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.cu new file mode 100644 index 0000000..56bdac7 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.cu @@ -0,0 +1,70 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_plus_pxy.h" + +__global__ void kernel_phxy_plus_pxy(float* __restrict__ phxy_memory, + float* __restrict__ pxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, + size_t pxy_dim_c0, size_t pxy_dim_c1, + size_t block_dim_c0, size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + size_t pattern_id = idx / block_dim_c0; + idx -= pattern_id * block_dim_c0; + size_t idx_h = idx / block_dim_c1; + idx -= idx_h * block_dim_c1; + size_t position_x = idx / block_dim_c2; + idx -= position_x * block_dim_c2; + size_t position_y = idx; + + size_t offset_h_temp = + pattern_id * phxy_dim_c0 + position_x * phxy_dim_c2 + position_y; + + size_t offset_h_sum = + pattern_id * pxy_dim_c0 + position_x * pxy_dim_c1 + position_y; + + phxy_memory[offset_h_temp + idx_h * phxy_dim_c1] += + pxy_memory[offset_h_sum]; + } +}; + +void occupancy_kernel_phxy_plus_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * h_dim * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_phxy_plus_pxy, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_plus_pxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.h new file mode 100644 index 0000000..3bacc94 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_plus_pxy.h @@ -0,0 +1,16 @@ +#ifndef KERNEL_PHXY_PLUS_PXY +#define KERNEL_PHXY_PLUS_PXY +#include +__global__ void kernel_phxy_plus_pxy(float* __restrict__ phxy_memory, + float* __restrict__ pxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, + size_t pxy_dim_c0, size_t pxy_dim_c1, + size_t block_dim_c0, size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx); + +void occupancy_kernel_phxy_plus_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); +#endif /* KERNEL_PHXY_PLUS_PXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.cu new file mode 100644 index 0000000..4a91337 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.cu @@ -0,0 +1,53 @@ +#include +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_times_phxy_equals_phxy.h" + +__global__ void kernel_phxy_times_phxy_equals_phxy( + float* __restrict__ phxy_memory_a, float* __restrict__ phxy_memory_b, + float* __restrict__ phxy_memory_out, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + phxy_memory_out[idx] = phxy_memory_a[idx] * phxy_memory_b[idx]; + } +}; + +void occupancy_kernel_phxy_times_phxy_equals_phxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * h_dim * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, + (void*)kernel_phxy_times_phxy_equals_phxy, 0, max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_times_phxy_equals_phxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.h new file mode 100644 index 0000000..b603c2c --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_phxy_equals_phxy.h @@ -0,0 +1,15 @@ +#ifndef KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY +#define KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY +#include + +__global__ void kernel_phxy_times_phxy_equals_phxy( + float* __restrict__ phxy_memory_a, float* __restrict__ phxy_memory_b, + float* __restrict__ phxy_memory_out, size_t max_idx); + +void occupancy_kernel_phxy_times_phxy_equals_phxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PHXY_TIMES_PHXY_EQUALS_PHXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.cu new file mode 100644 index 0000000..ed7c64e --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.cu @@ -0,0 +1,70 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_phxy_times_pxy.h" + +__global__ void kernel_phxy_times_pxy(float* __restrict__ phxy_memory, + float* __restrict__ pxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, + size_t pxy_dim_c0, size_t pxy_dim_c1, + size_t block_dim_c0, size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + size_t pattern_id = idx / block_dim_c0; + idx -= pattern_id * block_dim_c0; + size_t idx_h = idx / block_dim_c1; + idx -= idx_h * block_dim_c1; + size_t position_x = idx / block_dim_c2; + idx -= position_x * block_dim_c2; + size_t position_y = idx; + + size_t offset_h_temp = + pattern_id * phxy_dim_c0 + position_x * phxy_dim_c2 + position_y; + + size_t offset_h_sum = + pattern_id * pxy_dim_c0 + position_x * pxy_dim_c1 + position_y; + + phxy_memory[offset_h_temp + idx_h * phxy_dim_c1] *= + pxy_memory[offset_h_sum]; + } +}; + +void occupancy_kernel_phxy_times_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * h_dim * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_phxy_times_pxy, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_phxy_times_pxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.h new file mode 100644 index 0000000..bad68a9 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_phxy_times_pxy.h @@ -0,0 +1,17 @@ +#ifndef KERNEL_PHXY_TIMES_PXY +#define KERNEL_PHXY_TIMES_PXY +#include + +__global__ void kernel_phxy_times_pxy(float* __restrict__ phxy_memory, + float* __restrict__ pxy_memory, + size_t phxy_dim_c0, size_t phxy_dim_c1, + size_t phxy_dim_c2, size_t h_dim, + size_t pxy_dim_c0, size_t pxy_dim_c1, + size_t block_dim_c0, size_t block_dim_c1, + size_t block_dim_c2, size_t max_idx); + +void occupancy_kernel_phxy_times_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); +#endif /* KERNEL_PHXY_TIMES_PXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.cu new file mode 100644 index 0000000..92d5678 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.cu @@ -0,0 +1,50 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_pxy_plus_v.h" + +__global__ void kernel_pxy_plus_v(float* __restrict__ pxy_memory, float value, + size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + pxy_memory[idx] += value; + } +}; + +void occupancy_kernel_pxy_plus_v(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_pxy_plus_v, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_pxy_plus_v:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.h new file mode 100644 index 0000000..853a268 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_plus_v.h @@ -0,0 +1,12 @@ +#ifndef KERNEL_PXY_PLUS_V +#define KERNEL_PXY_PLUS_V +#include + +__global__ void kernel_pxy_plus_v(float* __restrict__ pxy_memory, float value, + size_t max_idx); + +void occupancy_kernel_pxy_plus_v(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); +#endif /* KERNEL_PXY_PLUS_V */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.cu new file mode 100644 index 0000000..beac382 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.cu @@ -0,0 +1,50 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_pxy_reciprocal.h" + +__global__ void kernel_pxy_reciprocal(float* __restrict__ pxy_memory, + size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + pxy_memory[idx] = 1.0 / pxy_memory[idx]; + } +}; + +void occupancy_kernel_pxy_reciprocal(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_pxy_reciprocal, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_pxy_reciprocal:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.h new file mode 100644 index 0000000..f943914 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_reciprocal.h @@ -0,0 +1,12 @@ +#ifndef KERNEL_PXY_RECIPROCAL +#define KERNEL_PXY_RECIPROCAL +#include +__global__ void kernel_pxy_reciprocal(float* __restrict__ pxy_memory, + size_t max_idx); + +void occupancy_kernel_pxy_reciprocal(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PXY_RECIPROCAL */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.cu new file mode 100644 index 0000000..4f7b640 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.cu @@ -0,0 +1,50 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_pxy_set_to_v.h" + +__global__ void kernel_pxy_set_to_v(float* __restrict__ pxy_memory, float value, + size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + pxy_memory[idx] = value; + } +}; + +void occupancy_kernel_pxy_set_to_v(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_pxy_set_to_v, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_pxy_set_to_v:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.h new file mode 100644 index 0000000..b2c31d2 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_set_to_v.h @@ -0,0 +1,13 @@ +#ifndef KERNEL_PXY_SET_TO_V +#define KERNEL_PXY_SET_TO_V +#include + +__global__ void kernel_pxy_set_to_v(float* __restrict__ pxy_memory, float value, + size_t max_idx); + +void occupancy_kernel_pxy_set_to_v(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PXY_SET_TO_V */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.cu new file mode 100644 index 0000000..1349a17 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.cu @@ -0,0 +1,58 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_pxy_time_pxy.h" + +// a *= b +__global__ void kernel_pxy_time_pxy(float* __restrict__ pxy_memory_a, + float* __restrict__ pxy_memory_b, + size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + pxy_memory_a[idx] *= pxy_memory_b[idx]; + } +}; + +void occupancy_kernel_pxy_time_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_pxy_time_pxy, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + size_t gpu_tuning_factor = 5; + + if ((gpu_tuning_factor > 0) && (gpu_tuning_factor < thread_block_size)) { + thread_block_size = int(gpu_tuning_factor); + } + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_pxy_time_pxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.h new file mode 100644 index 0000000..f2c67cb --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_time_pxy.h @@ -0,0 +1,15 @@ +#ifndef KERNEL_PXY_TIME_PXY +#define KERNEL_PXY_TIME_PXY + +#include + +__global__ void kernel_pxy_time_pxy(float* __restrict__ pxy_memory_a, + float* __restrict__ pxy_memory_b, + size_t max_idx); + +void occupancy_kernel_pxy_time_pxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PXY_TIME_PXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.cu new file mode 100644 index 0000000..c5bae95 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.cu @@ -0,0 +1,73 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_pxy_times_spike_selected_sxy.h" + +__global__ void kernel_pxy_times_spike_selected_sxy( + float* __restrict__ pxy_memory, float* __restrict__ sxy_memory, + int64_t* __restrict__ spike_memory, size_t spike_time, size_t spike_dim_c0, + size_t spike_dim_c1, size_t spike_dim_c2, size_t pxy_dim_c0, + size_t pxy_dim_c1, size_t sxy_dim_c0, size_t sxy_dim_c1, + size_t block_dim_c0, size_t block_dim_c1, size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + size_t pattern_id = idx / block_dim_c0; + idx -= pattern_id * block_dim_c0; + size_t position_x = idx / block_dim_c1; + idx -= position_x * block_dim_c1; + size_t position_y = idx; + + int64_t* spike = spike_memory + pattern_id * spike_dim_c0 + + spike_time * spike_dim_c1 + position_x * spike_dim_c2 + + position_y; + + if (*spike >= 0) { + pxy_memory[pattern_id * pxy_dim_c0 + position_x * pxy_dim_c1 + + position_y] *= + sxy_memory[*spike * sxy_dim_c0 + position_x * sxy_dim_c1 + + position_y]; + } else { + pxy_memory[pattern_id * pxy_dim_c0 + position_x * pxy_dim_c1 + + position_y] = 0; + } + } +}; + +void occupancy_kernel_pxy_times_spike_selected_sxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, + (void*)kernel_pxy_times_spike_selected_sxy, 0, max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_pxy_times_spike_selected_sxy:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; \ No newline at end of file diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.h new file mode 100644 index 0000000..4bc60d5 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_spike_selected_sxy.h @@ -0,0 +1,18 @@ +#ifndef KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY +#define KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY +#include + +__global__ void kernel_pxy_times_spike_selected_sxy( + float* __restrict__ pxy_memory, float* __restrict__ sxy_memory, + int64_t* __restrict__ spike_memory, size_t spike_time, size_t spike_dim_c0, + size_t spike_dim_c1, size_t spike_dim_c2, size_t pxy_dim_c0, + size_t pxy_dim_c1, size_t sxy_dim_c0, size_t sxy_dim_c1, + size_t block_dim_c0, size_t block_dim_c1, size_t max_idx); + +void occupancy_kernel_pxy_times_spike_selected_sxy(size_t dim_x, size_t dim_y, + size_t number_of_pattern, + size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PXY_TIMES_SPIKE_SELECTED_SXY */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.cu b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.cu new file mode 100644 index 0000000..1823f6f --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.cu @@ -0,0 +1,50 @@ +#include +#include + +#include "kernel_helper_functions.h" +#include "kernel_pxy_times_v.h" + +__global__ void kernel_pxy_times_v(float* __restrict__ pxy_memory, float value, + size_t max_idx) { + size_t idx = threadIdx.x + blockIdx.x * blockDim.x; + + if (idx < max_idx) { + pxy_memory[idx] *= value; + } +}; + +void occupancy_kernel_pxy_times_v(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug) { + size_t max_threadable_tasks; + cudaError_t status; + + int min_grid_size; + int thread_block_size; + int grid_size; + + max_threadable_tasks = number_of_pattern * dim_x * dim_y; + + status = cudaOccupancyMaxPotentialBlockSize( + &min_grid_size, &thread_block_size, (void*)kernel_pxy_times_v, 0, + max_threadable_tasks); + assert((status == cudaSuccess)); + + grid_size = + (max_threadable_tasks + thread_block_size - 1) / thread_block_size; + + output.resize(7); + output[0] = grid_size; + output[1] = 1; + output[2] = 1; + output[3] = thread_block_size; + output[4] = 1; + output[5] = 1; + output[6] = max_threadable_tasks; + + if (display_debug == true) { + std::cout << "kernel_pxy_times_v:" << std::endl; + kernel_debug_plot(output, display_debug); + } +}; diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.h b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.h new file mode 100644 index 0000000..611e0ec --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/kernel_pxy_times_v.h @@ -0,0 +1,13 @@ +#ifndef KERNEL_PXY_TIMES_V +#define KERNEL_PXY_TIMES_V +#include + +__global__ void kernel_pxy_times_v(float* __restrict__ pxy_memory, float value, + size_t max_idx); + +void occupancy_kernel_pxy_times_v(size_t dim_x, size_t dim_y, + size_t number_of_pattern, size_t h_dim, + std::vector& output, + bool display_debug); + +#endif /* KERNEL_PXY_TIMES_V */ diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/HDynamicCNNGPU.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/HDynamicCNNGPU.o new file mode 100644 index 0000000000000000000000000000000000000000..4ea2bada251dc7cdef3273b5900231418a7a2083 GIT binary patch literal 33080 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 literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/PyHDynamicCNNGPU.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/PyHDynamicCNNGPU.o new file mode 100644 index 0000000000000000000000000000000000000000..0381a0cdbe12dbda84a24e309a1d1040e606b1a0 GIT binary patch literal 367368 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 I$c`5OUlPp2$^ZZW literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_helper_functions.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_helper_functions.o new file mode 100644 index 0000000000000000000000000000000000000000..1ebde9c9429abe5dd2b0ee9170ac06b2a574d69e GIT binary patch literal 7840 zcmc&&Z)_Y#6`%8^@zn&|ZAvjDkSirn3SRkqvEx=sb&hlCo#F&5mxMH_n5^&i>?L=5 z$L^eSPJ|NCKh(7ns!Bks_Ctg~zz055LY1i06p23#;!8guA%qlykWeaADX5eRwIaMX z`<`#S-oqs=;+=M9<~Q%(ym|X}ZqMeY_x2@WTC8V-uf8w<-LE2E`X3(KOgoV!+Be!llQ@;&y4 zfk)?)LYzA@A=IXU7pJEFy7lC9S$o@uc|4wFq|CR??RTs$?(TP;yqe^dsS{1mtyp(`~O>A`iXiSoDf?3 zz&0^cIKOLUTv)mV`_YCTuk*VGVDhLIuMTW`6j|XR*5Vq(@yr{CUrLBU@s!piM2|=$ zP5|ECfU_Oyg-tjD!yH^6hwC=D;^?H~drnQxJGG|c%WBIjE4SeVR=E{&pAG9*wwHu^ z#3j)-6s5>+3Z5{eBexs@n-kzyPNXmqtYAJ+zOrYb$L(T09P3sF^IoMbEN{VTRh8v< z%3stY*Q+*!RdHrpbF%8&b%zHv*Mq*dAj3Hv%D@f+18b_bRBiQtHy;Ki=Dr`Wq@YbS z<{#4dMgs}Eb&c~bbf)iwZV%*xfIN7ozYp&EWdnilKEu-1g5w9U+vy2wL|Dr9=Ny#| z>T+USSazl2J3$}=_q3B9-@zg`-GW1~-nEn?=xk&P`Azx|4C3TpA*S`@Hv=D~sHs-u z$j}Kwkq1*cMdKlAbI?Y8!l{dKU<@zrcN{Cr<1=bflo#_NiXlG~15NA6WB$a+|CYe$ z4o7)0BQaa~dDljbTa)_Sa{N%w&B-Ma=RkIX^o~QD)BTjLYbT(u>snfaD>vpZ0D@0I zdU$xw@f_b)&M+2=VXP`?A=ayVoBI;t`d#J|(zogf>5D!geL!HZR5%|V(&&^x^}BHX zBQOb-|H|>Ol+XEYCEdP1Ep%s*JEx(e=6%Pmq-Sf5@=49I3P9FX0E*P@wsb3~?uDt@ zz0tix%WJ4yv%R@ASAfizHDZmY@7e2A8or$lhr=1GTF&nL9LnB73KFLeibQ)bv1LM1aU@!m>vFXIEcyMJ(Z|IlQA`jHJ6lHcs`_o$IQIg$>?PUMp zSNl`CnsFX&qIt9tz7d7^Eb%_lV*~eprZ>UI4dQP%;<%Er6#?r^f*=Pvkfy5RVqZm<(h;0oz0 zUF_l!7wKNOFxTF_5favFBvgg(3&!t85D1?<{EN(Z2012_GAm++{cNpuhh8?2V>C+B`Np4*5&>gNFiH~pM5 zaMRC*ft!9V5FYjOYlLIdXZZP~p~sBpvj%R)^Rj`Pe*Ve8O+Q~ZaMRE0gh%~+n{aIK zY|ruB0Cfy08qduJZu+^~z)e4o8@TCb*}zRdFBrI~=LZIE>iLO*n|gj~;Gcwb@Vb6M zIIb6K9pk??__rDOYX(0a9WfqAzl-4fKD`pb`MmZQLw_3N+3uSLewTqK$q@Q!+U0XU zPJhbaXTO-A^|N0%f49Mp|HF_lA3hBi^W&Z(p`Pt0##53=@iTPOjhofr`(F;Ni?2E4F^h%v1+Xq z%+p9sxQzf@2%H^7;MtRj$EOev*P$Hv90Q(a#-yy;YLBzvmYqW(SW-L6jh3Fyj!Fq% zc7S8n_2zA_QgdXr;mf-1*>g@s`bt>kM!gQ7iAb@x5Qehk%xM%>+O{S9-)p<5v81La zq;wm&6k*NcSE6bB5@omrL)%xarjQR#e{uKZG=#D>D~nRqo9(KSBO@dBSk}1@JDTLU z?Ed$$VppAvRc3CXOGt!Z7Gn;Knz=Z^y2~wgir^ie*TLOnKh+rSff81+aJd zgMejWU)(Zt7Z1rpBQjqamoxiHSy_XxWDVt54<0x?4C^kR6c!{&UE&9yz?}nI$ELFx zy$P5xc~7}&k@IQFuGMBiW@cs~5BumlRSKgsPoLCiBLns%Gn`pX#LDb?BW@+ zKg|#nm^uenhv@ldew0Uw`u_lIT-0~uo5x@S{g5^mtp->25C~? A7XSbN literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_fill_with_h.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_fill_with_h.o new file mode 100644 index 0000000000000000000000000000000000000000..9c19be088d1b6b3284f200ed886000addbcf3134 GIT binary patch literal 21000 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 B`iTGl literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_fill_with_spike_selected_w.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_fill_with_spike_selected_w.o new file mode 100644 index 0000000000000000000000000000000000000000..92dd0380dd5406239e01f7f56b172c5bdf41cc59 GIT binary patch literal 24816 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 literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_plus_phxy.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_plus_phxy.o new file mode 100644 index 0000000000000000000000000000000000000000..9b82ae4e7dd343285dbc0ae00b95c559230009ae GIT binary patch literal 14744 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 Q_Wv2QQz1UGp!K}NF2xoxJ&AO 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 Q+!XaXn}?APZ79_Lf6oLalK=n! literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_times_pxy.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_phxy_times_pxy.o new file mode 100644 index 0000000000000000000000000000000000000000..a50fd9c23a10aee0694e0b598ddb0435e3dc4324 GIT binary patch literal 22040 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 WzoP57j3wFsr+}wId}KqV{{IbYKKbAP literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_plus_v.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_plus_v.o new file mode 100644 index 0000000000000000000000000000000000000000..7a14bcdacc4725b5c86c56ca361d4e5b1efd969f GIT binary patch literal 14272 zcmeHOdvF`ad0#vvkq8BVvMf^%TaXOOq(y-M-=rjmAVKM1LlVc7EJwC{Isyk0V-P@q z0|m))L|aliR|cu3(`K9|)4G{@>S>#}naOlqPoq+nCvuc&s@UnocH&1|r;#lyru>rR z7yEsCyWsK&*vj}Hccz~Px4XanUi-McZuXw>PZ+Qqtd(^be)tA+H_l$gwHcu`5M2~r;;HBRLF2pJMT&s_wRTAueF{r;Z;EKG54!D^!A^q*ZPX_{1C4XLn z7J)8z(Dm3)Lfw~xt{r~}xprI*bszH9Xf?s2ebxHAz0$F%!;eLxA4`*9Ib5_iOEXVP z`pHQ373t*bo(TBat2qMCgfCr^^i$^-1)d3Bx}={9Pujg<=w> zU^}r0yQT5x5&ien4`f8X^EUubSAHSQFOiB;m(+a*3)+G(4z3janWR^_8Wv#tT(!M0 z+J!nuJ(51^Kl+U8C{7@0@_MY6inQ~eSb}-|Wok#7v`Le_=cLK++NKdc_G{Ns8;DHY z*G062u7Bz8Kfg@m)nBGlVP??pIx#IxZbrR^BKJ<5y-V9hSOJU!Vrg zJEfDqvl$AF+9dKG@_J@xqBIFb>DY|6!0_#Xr!c)Mcz1Ab@SdQ&_mlw_sr#&3q~c=q z?FkP&XtlwkpE4_BJ?C0Yq$6*RzT>KeDJSWgSCaHY-d?E)ZyWkQh&iYTdGEv=)P}tK zq{s1X!5;=H;D$mSEnr)uq6QPRxYOGu6(e3h_2t-@a)F^nQc|9-*8l=gH z*ByenTE~VW0?dOmn%57%UZVj=o^n&6|C|28$cl4}PIE~Y6P#Tz1a4QY-CJ5-=upDE zk@Pdc$8nYo1b%9Chd5=6n{io#-f4IpQgWPiA}t^=cwyHQ7UCv#OR+?Vl~^oyLV6C@ zHGvkvh~5F^UicwvE&R&Sy=pe4Cgn`wfSgI@bMm+^nn%9F#=75%YXFj!oFD}wOP{^W z-b9T7-m%~gu+>Fu@O0s9^5sTzT2@nrA;;4LV@8>#a&9ygD-d65+?S7QJ~gFf571H~ z6;BhNrWQ1xyeH5(&%?oZe5}Oy87#R`C9B3hgDIL$rVi&vhe+2gAZA6sL>HoVpe8O4-Q5IT#kUP@KP| zv(>$jUE-egX9AaTAY6F7MwaHWY$NtVn7V2nn3!lFRWlDvG*r>7!QaOC%?a0O$7a~; z;9)4BS;qVikXp(8@U&PybUNTkgGozsOwklM0LgeZtu;qeDLIu+p|3XkPI z+X$3w%i%FSJ^hZMi`(;!tJz#4o$_?~{IIMk*%4LqY%m&Zy$Zn=5F}%1_=Fh^0%P@BBIYY>6j5-!J~bZ(1NBi& zVeCrGmrVO|hMqR4%3^&MWwU{eBs#8PS8ziyw3KZF>p4}+_(oLLPZ^3EwXxQXq_;j3 z@C;`aPqjB9%YA)*xdpoyv?)F!jpy4}P^ z+Eu~Nexs3xEU_;U^0h*~(@x_S3+yxOjw)D-2-#<;Bevyr#D1L-yjUTgBGYS1WeaMb zt}T^OKi1n;&Jg>mDzXoXXAuMXQzM~z(XZ=G<&de&ivOQ{!F#|vZkZwQ-IyWYTtuF7 zk>v-N!)qXIa6;K;0QHb~oy2<(KF>&2;6JmP)EtET-GpTJP20%RE-cTGy?`lDe!)ey zRU?Uke^^ZsQ(JoFLe=e#zh`DK&<%L zCChK9A%Ev2YsMjFz#s(a%ntH(s0W1Dm%D3F-!|?)v$qw>)A-S!IbtJIE>c&j2ca)O z8JqEW2*VvVLUYvgveF(F5AjM30|VU*LwWo$wfH5Wv=|yxPQI&*JVy0q`OXURQW<&l-7NnoVA!YTId~_NF9;o8 zuKm4Ce%_7)UksoLCXa20`CB!Cpg<^xb^M$_sG?c)4~eJTe^iTd+)v~Yse#QKo|@(U zczYHLo@?x59d!?+z0_b8JwMq8d3t{0pIOU)*np=P zo(kI%;J?OUA#bk|_XdtRdjp4gZ_rF;>aTWZ;1G5OH<3lPay@B>9Y8brGDKO;#MvBZ zx`(WIr>e!#=(x`zgzZMu4u`Q{B?B0**An+|E$M(by_QTs)CbYrrp_hg)nqza9q@E| z{0@gB8T%8m)|X#Pc3`|$B#S%zdpC9w>mjl_&`I2zA+}DF)vaB`UEi|zN91q2J%Kg{ z4ViyS{(d%IzLxC8pfI(JlFq zbsy|{*J8-PiNEGNotd$?$p`R;eXzjw>_Vb1jB!pQ75-jxO{~lT!Ed zt_NQO{gS@Z>kfCnEa}gOU0eTid(qn+?tUub+WJbwawX#Gdu7|P*R02Q5G^Qpkpt;L$l68&$(joXeK(PS7IZDFa7c_T7pON%GNN|kpHZ&i*puryv-lBzV`YlY@ z{#;n!QgvI={(DIAB)d*H#upEV!SIk&q+5d0-x7?&Ua5F3bxVKguGcX`8sG#{%jZ0s zH+!P_m@<@)M&V$fvHY>dcsddDq@&S%MoC2v$n%aq8t^D=^(C73A zon0a4;E_e&SXf-}P5YnOzHYTOSe#u!r=QjX7x21+sCW=<9%s5f%_jQ%IO9uW8glxr zac33A5N8?Aj@M~ih_4XRP51!+j#w3E6=Zq)EC&Z8wDpU=gB&Y)9}GePeL@<8@0J}i z^V_%8`LK0P1D&?=4SX2Bk=R9O8^a0AW4z190J8Ha2QqeYab{ix05$@ZWeCnC=b`&C zI4FaEL~;6rzd7KukeZSO93!MF zFz$%s+(P2`Ak|Y-YCp%1Gk^ZwU?<$4`=E^e^wRdhi7%_i#6~x(0sz_d(9c!@r*jwZ z?=ica8|M%w+Z?SNMg175&L#9?30UBS^Bd`JpdWZ!4a9f`D&TK{3e=Nf^7Oq0k$1td za(aGT0S|%v+EVWlLe~<|gNt7|Jw@Ozho1(Sa`^vL@c&!|{@0;h<@7XG(1V8t<@l#6 z=>Kj7JXN7xp$h!CFD+*m?{$>J4^`0r!wUWUbOrs@pa;j@1NBIJ+*d-roZWw_;LrCf z=>JXye*iA4-&ukG{R;eXv>Y0TkE&n81I7fVW_W0-5>0%y6{38h%{QX#w{L z_(u%KE`=^B@vmfkjQZCI_*#Zzf?olyznS4^cddZmCEzy-IDQ|I%=!xgzZdGc{s{rE z7w{waGxt;|u|B=wo2KaG&seogE#qsq5j=pldh2dysk$`s#IDT6=f3JXJ z48ieR1^w#*|7!vF3HT=rH_sn8b_@_E&G6%TCIwv7^8<#P z^-Kx;qMlzd9Q$p#p#L`lf2)B1x4@6zd{!iHm~j4GzG%XEjAqBl4P^H7OAJSUaFyWg zUCnS**CycW1b)$<^#VWq>7}G+qkwk^cz*>wlE9B=g4~~PKpqMG5$CzMKKP-I^S_JC zK*%rR)xtU;z8A{`T%1=SVSQK(y14$Zz#kBB*8T8>F=9;4&&LPry99o|GUK>(3pkIz zP*0D5-w1gm)Uy$OTtAQF&@DE#Ava5iqb?p-iu^q0o#n>Zk&w*-;9+^xfj=HMNdnH> zh|pgMIP!Z~$NZZK=l=ZCg!6dwoPdjdx>*0C?oEsb6R&_{9j6Tf?%*J#*Kc+%UAE!6 zo;5Bzk!9S9?@Ga+wc$>Jn%$ykeTkF;*OZK!+vnkU8A$OZQVG}?XR``0?itbo9kQ&A zX4CuSVI?LgFn-=Ho7ILj2L2D z@~P-nRSRlbHZh#nRJgwt!8>e1Q=zD2wS0#7MzU#0|#n>4fxI!Wu{y|VHk-~4-WCOu+lSJvnY=a-op)1H7-r*2 zVBQ4onMi+?AqSHGrVKFEW>5Wmd;N(q#Of+p)yU5VPY%}A-Em} zU6sabV!mh&7E$&V`r3G=m1kObCQy=qSrYJJk=~v}9iiP&i2G!Dq)?DEa3w39Qj##P z^0=RD?eCw{w{TA@(mxpX%NQwqK6)FzbLGCjAL0idXo2bIh<%ofN_egZ05KCA>17-d z79X+VyglVQ5XMg}=jVL{6(6=U>C@|$L3|87Qw%A_RlhTlX==!L$mSpg5$L!!18stc?9Ay3bx+^Kh_fsO5AT=#{Cb~X_9}a!Qgn5F`|9!JKlf1|CfOrwa@r?(uGTW~Od9>dvGGQ5mTA+fF(DOZvgUt5vKL;RYp2MTezK6;4gv;~tkJ$LPaDb&; z{&^FQZAIegQOIE4Y+u4d1R&l|Mrqc6UOo)M<=X!^Yd@FA{>Rfr?mx%77;d(|kJ&f( zH<#!3aqm{n{#j<<&lu4@>f-X;ex>>AEi->H?xk~0NS|i?Z!^um4D+Ac0})E*@+pv~ zI!*F);|LPmPxK%6E?l1D<81tRJE;+P6qtS7n~Q{Fjl@&juk*Z7V>};W@*;+LGwp3bKI!?(^yHNxI>5 zcYXIHpMk^z+nHR1Ql?2eqrmJ864~;Hj|SmW8)n&>299v zX3IdiFM50Qw&=d--hSnQSEf-ADIZ*Vk00%>h7rW7*ybY z{9N4s_$$ZGPsBotqUG~3e{4F|H68OOrhybhGB%M8E$NwDzOM32`MJ+VFTULysjU6( z#||syY^W1-%Td>cL^&N&q8EP{IX}ol*uUZ^X=Tx_=cDBx_4*TUkO2WfiJ8n19cuSl9ib2BRT5>0W-r-67@J)Zxbysb4D-&>Ww1 zZ%`_ymGX(i!pq8up9K@Jv)k}Sz7xNAQ7NB1S0DLK^y0vG@YY$Jl&*BDxFH#8Oj(7;?%fG>KUc%@i#A^ z`uQ8X(b}aYh=WRbxaH_~{70z+DHGRHu`|I}himtWv|lnIwI(LJq9e8F{r;A8*p zKk5R@_LNrF<$; zep4y`fGarXQ%?NYWePM)vqXD;C^*v-l?h~(V-;h8Y1;!&Q+!wS?&$vL??l!8Crx;g zn=jm=OkPd4Jr%@*7n?NYlXhXkbN&@fIr7RKO4$f?D&_lO<|L{|mUmMMSS+L{Ptxe1 zA7uk@z!Ed#4k7K~Y2FpfKT#(4Qauu(AXj{B%2=QrdHN2hH$nll_!4EJArwQu^a!^n zK-*n0LP7jmOkGYQa%%X zl6p~ZQMw+(Zk>;O~J02E};~4__&hLB5!2;}&rIcgL$sCRYh=*x}SVBEo z@HXVT@mqplBYyLRdo(?#Wz|CIkg98`OhM15lG$)-RMH+c=l3=m>lB>qI4hY+{+xQh z$TXSpZ3}LL&Q9jS(@(u9TpTVMx{(|}O2&sq%sh|P;&3iqV&U9acy!PRYdJ$d#B-V4 zV4j5ytz?ANJ6j|3TG}@_Ql))os^V}`*V6Aym&)ggMlxr#yaQ=k8yFo@2leEL#=>SM zo5>-qYuQ=(+?YC3tQM2SqDeC!Ud0MVNi|A_Rhm!*@kt@yM1E5*U)jI^(@hY%k~tw& zO^v3LYH_5sMb%RURnMt|ZLMlCn?G1c8pFIhyvqhu3rVz{Y`Xsq?PwWUvy=Ogbur&; z5V>#Rw+i?ZG%q9cxy`r%c^BhzjtPk8O&k|QmXe(u=aZNPWo(G!b+a(ae468OR+uU7 zMzSeju~V?pgclNIl&0Wf?)xeBH#{&t-po9e2gaK{Jjod0?K5I$rDR>*R1G)HZl(Zt zVupO7x>`o&Zda7hWjPZ&H)eCq^Q5V6!ReeHf01q+GcL1N+_-tu|zF6e<;m$27f|$>)Nd;TFv4NqtB&g2fTFvx9|`>9npDi)t}*uNLg!jUFD$ z6f*-^Eoi)Xlnpv0+<6wBSqy37M=5SYA*1cDb2wThUqj`8J7>3i+!G&uITU&41 zA|OmZKnAzARgnRX=vz0=Ah2!W8a=&r6Ci|a+O(P*O$Dv%f_Gi1n$r0FTjuh%y*u6r zc-crE?^)Bl$hzPfBa;r7wpdXZvL-hdhK@BUBPqy}>2Nk5E}C-MQKVD3PP}?#3rmd| zR57V2O`7y}XfJ9;Av~msS{FEOmZsabu0{|Yy z{O!JO<~{H!PDFDS(UCxZW(8}w7x0 zd)dz#??s=y1Ns8v_3R}-`+Lf72LWx1zCqpK%*dYAy30iryd3(XkY@%{-p5g2DswLlG?2bsw(=0_OZ>dOP~Lk8a=9?@ zPow=x&I+xVoQq9;4La(Ne4%fq{*NLr>px2#{9UEL)udnTp#|Q!%6$NRo+DhDIq6+M z-c`@|5>VFP?L>C`YkcKM^>^Cp?{wCa{bo7*;URa-$S3~>i9obZpyYC*1uACy^7f>& zAZBgP7YPvbO{J@`raR1d`@`Ol(SUWK@E7`5Q>hM%=JA^=Z=*fW!e9DT4ZNHxfxM?^ ztB*kru+&HQcxJcf!g`v!>$~xI$-jKH?lJS@$fLcRO7#n?_MG**uhQ8ju%?x{>+yg@(ZdHr8@^Hb{oJecJn9nFuti{`VlH^W{Gf8@P9PyKIvlQO^m0XlNYqv4@!END4p1&n}a2#Z6E21v9EX?X+}oXUB-R6R$A}e{Dw9 zr=4~I06Rv+FAM&CRZcTur;XI$znM?UjJ<7hj-7v#DHel&cGxQ3+j|8p%)uluJEY9) z9R7#r;ipprJO3NN1nDfn(*iWdK4YYxezN@}z7!RjkN?s<{Nxfle@_j553NTKTFi*+ zCfn`&Uj{}#wYiyf3jTt{;q%qMTkwnKLQVv8sn5)pYqjoyaak^6;v>IQ&uMMtkBPjD zrTt_U@ku*vZ0xi+`jN4n@4f4DIL4u*PdMrp#NAw}ho55~K$^y6V-OEZ_+v4S6<~A( z9$deENXuz@($Ln^S?qc`MGju!m_*2WpVz@&TW&q3@Xk7>@Y;_lHn2+bdpM-c%khGFy3oXq(N2>N78+S#ppkXpK%o&|Qrl^VyLD3& zdnubwEsO*=1zWscZ#Ml&wkkZjitVAJgs-#4_4~JMVJon$e&3qN{v+&*UBO7Zmv8@{ zV1G_KZ*!CXezu=)`4_XI*yCTrepl@8KfqSEuUo+!n^^(ddM`HOf5KdwSs*>gu%AE5 zz8;AL+q_~^e}-+uhQ66yUWmQCx4M_FZjHBizs_Eoy_fIB`Zrx>*QZ-E%xDd6^kOIf zkYmxtLH1zl7BAnvU&F5NSa&09+^9KRw=?ID9o@83zn*=1{}!*gQAf8p&xWx@_ZE#7 z=DdO7!!r8{TQJ4mzxgxl)ep4?BVO8*d%Yj^dc%X6tOlR0dKU+A0d}Eo-gCGxmIt%X z;Wjimod~7CCxV3I7xD7PJm+$e zsaqx+E-Cb__m)%cOH@XT@$xAHUj*rE;A9EESwiAn=Z!$3Tv5tDSMcR9U`jrA-oQuM zU+~PX@=b!-{an1f-LrGj{ZqvB^>B&hGpY6C92EB}ll&;7`mNUd*rrVK?^a6rM|b^< z>|@RN;A4yygE!t7q|Xceqp1|mB-hqHULDJ4(!qQxHCjmKQis%ek71kXNK-Y|1#8Ae zdh_NnEoJ2Ot)sbOW+;cl#cVz|bW`y9b-{srK3l_r&CKN5yjk6aPfLTz(X62+3k5Bg zW>-2zWMA$X*e?4bulPGXQrTo~C@7C_BOAjl;f`SQb}gO9Ih~nZ9|;ep+BU5vQHz8; z$Gx{QM=9V~)==+x7z#tcmy5E`QXKOga31h^qP_zIzTud!5ceI3`8-UaB0;x{42}8% z+k8v5`GVVg&D(q}QD0}w*LS4;p}NThpK|}6>*G#Wv%}XJ^|kPFP=PiOCBc29^I;+D zN;=8rCk0)e;+U_+Iq37y8sb@j%hxTvgJ>vG_#i!C-w|gLme17Bac^IO*S`MqM6pcz zpbr_ciNXlpTW(6st=|scqt00s^ttA*;G=j)l8Z<;EoUfA>s{FfNX`id!Q*DH(w1jV zaLrkzm?ZdaGKcmPn8N0+))>2*FI0HpES|psfM^~5P$)z#Vz!(Rybv~HvD0W)HEBc_v;Cxk0_KLKX0 zuW99G25Ij&=p3w}s)5>wLY=^=Z&8qV9~zJ7CBMYKDC~L0Bxmwm_CDmvo*=IuzJQ5x zid_7#%mN^J1N`i?;Johw|B}!vsi_ai;F={>q9lK&z7pL9JDQ+D2I}7w{u1?-bu|(5 zc~cF1E%-=|+@JAr3zlEPu=(WtLk;}Hkh7}lUB>ua0y#AK%_rw`pq~#v1)llv*K626 zU4#BrxMn^%9W~_m;m`T#r)tRmLJj=B8ug0Rpx;-+o@Z*FZ*q!%{X!ynX;^8(s;rITiBGsyA& zlN#;*Awfs(A)mZWM^Fg87M$L%50M>P@3P>l zEqGDjcKQ4>0_aFi$U;AEk+a5vPg(HmE%<*}aGHaqoeMCID3JVS{3O1}g3|&^;x|}u zvQ^@(0w*=~7QD-XQ(u$x-4>kI5E8$|A|E@dD!#{pue0Du3vRVDW5L&3=ucblumyiv z;P(FWwuOF!g?^YEhCq>3&L;(Km-7V+y;aWN37q^!YYpj#uUhEaEci(aJ-xF*Tktv>BnV``)t;*aPW1R>T$QuTf_GT()ivZa zTj=RbP}=hl(iBL(wVzw(hZdAc`d<<;0?}LXg_e21Ixgt500pAA_N$m>exTK(lpnXy z(;89Y#7{x`N8+-5sJs(DiOZRp+NH~a%k?11*=)gABTa$iY{5^;k@Gs4CAu(DGl6iD zCD)Z!dcH-SA>}EeztMsPMVe&LPp+F33oh$O&}S?-(aZPkS8ceo=X*9>t~aMGxYbU- z@IT4kDrhJQS#TG44yMxMvEAflO{VnM-Fibuh21BgtHHNK30aZk?v`BO|zsD+K4p zOt1HALmY)uN=X%`tfdS|>^BndLMBgDVI$fIjs#hF0M}f}b$jF~$*yE6nldtD+BO_q z-c#(;^j$c&;FpeZ*pk<8<nov&kWoZTo00wL>$ahM{K$Mhy*jW)pOW*iuzwle#fl zVBsM>k4W8-*t0E)yFiAxDZ;RU7-=X&H z>~B-E@Icv24#dG2blsU~i|}qst<;MvYQ`ZsSjn9O;OXld z>qXDkwL$8HIP2E9(dIC)_9dq7%N@+*((0{Rag=Uv^9e&67MHo<_hMc(2L?1JaMhSk z0vH25qZl#@A2R55sg$afQd+@KaicwJYJ(nAVXf(@mmkt$;(%0hLNcccRnzsn-nTOa zmAm`%ZG};zH#O(lyn1J3lQ?ZO@01s^d1Fp{;<`PCLEo*td-r}s-M(*k&))c+-KvTU z?QM~FIJF!>Ky%qT$N)2Eix}>&!d-c~#~Mx*F{_GU8E%(}HkoLZiAa?LJtY#REWg4| zGGezQGZ`q)v4w~ zFubvvn@Nm|1(jU8Q2m4WNq!mcMB|ZuerW}<41GesJok~fv|Hx4gP+T?JCHUL2#=W& z_doeg`cL|QvGBjW{T~gQjI@@u2D|=7@RR;-D-qI<3H>F3Q~hauWYqZ}+$42OAHxr<+fIr_kRb=tw`wlKfJCt^WJaYLn60e`!6; z`Z3e~zE7;6u0l{y*A3;*2{Ze8Y{8ayL35a-H)c-k= zr}m?74w7Hi{}SwXB`SFojpJ7hcZi0k`$rUrpMu@~didXNJJCHN+W%RUQy@N)Vb}j( D4W+|K literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_set_to_v.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_set_to_v.o new file mode 100644 index 0000000000000000000000000000000000000000..f0aa1dede2fb853e017aeb5623d67981deab99f5 GIT binary patch literal 14264 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 Hx&8kGLn;7U literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_times_spike_selected_sxy.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_times_spike_selected_sxy.o new file mode 100644 index 0000000000000000000000000000000000000000..e68b6bb6743bcabb192f8ecb6e631ef3b0d61448 GIT binary patch literal 23088 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 Ht^WTD2Fk3A literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_times_v.o b/network/h_dynamic_cnn_gpu_cpp_v2_pre/o/kernel_pxy_times_v.o new file mode 100644 index 0000000000000000000000000000000000000000..c9673845272c13b41bdc0a6343c01d16b34c51d2 GIT binary patch literal 14280 zcmeHOdvF}ZneTnrmN#N|WsH%rgU4p=NU^or)ysO=L~E_=S-iGYNCtv&#-rU?X^(wn zcUE2-gGFrTZc15>R3)igg`$$XOT|@J7s5Yxgt~Ldn8Soz?tDO12nmmHq+&1*G6b;0 zL-&0>-K*AU7r4qlT-AN5neP7e_v**=eAC-HIvCw#ni-5RbdY#Uymyh_ISVvAwdrn-j6EZd8 zb0WI>)E5A)RJ|xvHJ);bRcC~%<)_fYiquJ=YUGqxtSUp5Jk<#hL36iNWOS_XJai%A zdFYK37pKC$o zjg+q&_*O$(>!U{8Cx_nqP4HrZlHarDI9es4o)<#Ki-Vr%JLrIyITJSC3jS;`I9>7Q zRcI0D@`gMQJrVA?9P;e>L)f$Da=7P+uU@YYm7S|M-rp{SAY6*5r^M6E*-%~%35&14$4?Nw)#WcT6F3a6=&r?{?7e#Pz<=D?NOP^XKA`?ssmbh2e$BqX!ASr=?@*A zTc2@0B09GE3>G+;gUosThF~1n0xPCb_)nJdSHXHqJ(kcV**w zYT$xfKK(m~snD!VA@32NG&d9FDJaS(X7xp;Z}&cq>Auh%q5Yw+gp~bfOt?(lXWb%~ zuR`A*mEcCJO&;TnRT=9!&l(~hdt>}^mSof=>%xSd^ecns~trg+^Myw#a22CTMxY*DaT$z5w;*#I(LhKKA%Hc%=Izm;_Do zRMh7U!(44-LlFh$p;_G*fUDnZz_G`@ROtVvzc8}m9HY}*HpB#H7Yu>d)8O<~mKQpd zFmGhzsn8=h%Lao#GrL2avgNI~tU>QIy#^_{pLHTFATV@s@1r*2C5_9mM2MZ(Y_~yr z9@jO2mcWSK0_A?Vu7Rrou4;6rmd|J@C09D6=*hHJP$vAbA}}5`*Zp=}1CZ?G6e*cm zdhciUChJY`mPNOKt#0Cgy9Z~Jzc60V^SU|;Ii4O$n`N3Rh4D^>+uhVC*B^UnD*34BPn0Qt-;&12a?oJTPA)ZaTf*+lAQ@11O_qgj)`gYjDvuG5Ll za5lotR6w(g`5z#)lY8K9v%T+j!JP(`w$`|+t4a`(iF{UXjb$=QCYvm1t?5j%peqBr zcem0quw*p-U^ahe>$nm>lu^^kn1bA6xuO!w<&=pwC70I-S-4!_f@Zt_`DY$n{5q_y zOCho|$!T*ewq0`A?E#y^hBv$YWn#1c#0JjEz+#7G*6J<*f&${W+UBUYdu^9y-U1bp z?H$J$lpHJJHZwExwyDds1}ox(mMG8NGsi4#IMHVd96@T3dzHo z)I~emKangXM^l<)E{#&{XSRv>=>sD9i(MTb!)HE5N?=okKhfUy$^5}N{!gX9jWm~s34;h$#3bLL!cc?*XQF~HUk8dikmH@4aADJ zP13&F9l{&Bzi&>GnGJp#!JsIJZAO#rpGL9`BuP38uQanwU~avhjQdM{A`0HG&n$$& z!upu5GIllYPi6fDQ%?s}<*`1GviV>iiB0I(8{AMFE#*7FdO_23{xOYpRgU6jZM?mY z^smnarO~`9E%ikeW!tuZ(uO?@+El-u^^c{pqiU+s&D3lhISS(`|0wH;PBeuop#mnZ zzrgS=ikmW>wCkBV*7rDQAlLL@uvH0YRYB)2&{~rn>%etV@wk_eyru(k zH*vPs5nT1^;Bwwz<{?X*%Y=M`kl*a2af}VVJDjdMc*qGkU~43fm5s!Ctr^VF2Y+U5 zrEEj(GYyq8>L*a|ST#$WtLw-CD4s(MXv$2eUi6E8gN|KC4rJ>Z+L z%@TNB%#!<-kjFh_V^-a~dRMP{T)2YKB?8Vr7hPhdT$|1#7s z_Ym@whunLJ*zt8sR$gCE{@zWb!w@UnjzB-u?;&4@dO(PCrMDjS?c)A3dpn>!gD?Bp zV-7OyA&r%K5c&+1!HYkOFw9{iG*3@ID}9Bg!C+2R-X`mnaSs{Adr>!w@_!rkbQ_w8bTftWHD>y{jds#e6vhG ziUonJ^34EPF=XF~lM-mZrE{&kn@u)rGs#KTQ312Q5+O6U7z&SB4f*GoJUp6|&>f79 zt@0NEt30BV)4X*v(91BC$BU02#~YwQ)#T+W@)*rq<@;*L-&RE)eYeVEyp4ToorB{{ zzJ!&GsX2o@mj*rF%O$W%Tx9i$m&91v#La1&=>dPSBBZC;DHxTb=1L=a;yn#$Yml-gY^@_mwYLe$$tt}KOs*HsC4E74=0>oM}m{D60mtj)&_vLW74I=~+P1tiLB7`3=b|z0wPZurrmdu*GiGz#O6(8V$g?)GfsE|$bES*dkag{wt|KLk zVO<5ijo4cV{QNNp_QBXSz2701zStoJT^O~xTz9)%{zNjRK@YC`6yLvIGPAhq=dYi9 zMCO0~wl1?DfY04^7&Mr~A9J40&f2`>Pw<5OX_4od#l%<~=bUCL{9beHjYj#zyPo5# zKvp*1ju;17{831e$x>( zuE@dDa?h_kUwakw%f^V$8|isTHhvZH?D)^!WnWLE=kcg#$IDUMm8fUi%ezjzs)ueW zZ}}ZMa~^LOIwEHWF+(EdyX)j*@7wh2ZxpI>_%zBCTTRAO;xkqOH7g8US${Ta3<7ZN>|iePRqne?ay&+ z-@^l=+xZ;C@+VR(rDnz?zGV{Z^auQ1Qqy)Vo`sKBv)B^!Ct~fJn^70r4kxz5HxpaQ zYg<*nr0#w&=!3e|%-BC-8g}RG1$SM@or}3kVfUmh;?9NLbuv~lzFp{O$nD+cZrtXU zwz-?OxdS10ci262Y{@qkmlxgV{ATU57EmIfk?Cbs87qBZTw?yn%nm?5evCvb=q^!$VQp`X%2(j#a!5hM<5x zA*JEH<;2Xw_U&+g-#)K_Zb$V7ejnbE*hOd?!wJk|yvxS`vI_`*P<4_^vkNK!a1f}h zLa-mX06l`iK^5Fa0~K01Pv%De*~nGoTMXqsx@a4&s_^o-`zMT_m+{?>x%6lE|CXl?Blgm2y;~wK*jZGW{nRbRA1EPPN0|Io1_)o&XkvPPYmB%-r9Dz-J@% z6$>~D&eUZ<$@rjdO^TW1d!yqJE52=TjK`SOyk2;rvGWJLm`ARueJroi*@FpaS*e zm^^)MLFA9%ST#NWS_79seqE(^388BV=)uLWnx1b0e>MCp$W+5$ui^js8vGZaUDfoo z)X=jS`m-AUbPfFv)WGvK+7+(BAFbifzt!-6vWEVLYxMKK*U(=NdT`t&s7K=Cj`O`* zKm53cKM&W?|GgUi09;kSy9Pi0PE^gGzd*a-la;<*L(elXzH9BcTARe5_vdSj`xhA> zb`SQ+2e@SfG$`Qs{^9t68aS>SNT^5T{|-eL;y+|K+Q$Hj>*xCxsONeCZ^I4+vhsh4 z;mE%RF0SXSfJ*}Y0mHFNp-U?Kt5_eS{H z68L=r|D=Gg7w~BT-yq<>6>waGxu1(*{v)BEO>l91xqxGU#qo^-j=pldjp1l!iGcSA zILs%;Rf4y74Z~4ghk$Po_(gv<3jFX>OGQtgfOiS_Kn*>zz>j-^+@Ehi9tr&s z=ef8(1fY)d{}GvikYB`?3hRLQT&xgqabAUm^#P+tu0JC1<55VCv+l2q3&;8RV12j1 z&sSy~mmUG<@gR;%uYlhGc_h@+2N&1R*L8G@O>M}{5#p$e$CV;K{WP6pr8(r^Dqs@J zqYk`y+$0M)ZzDqA5^&^~SjYT}1?T=eYr%QEd0xOpKRvAfQTGF2k)m#=8Kws)<^NfZS?#rJ05muxtTpyju#`nF_7g=0!)&CLsNJP4%vlbIxZ73cFR zFy20_2fGwSAJ1nGDx+##iKzvB0`O>ChoThpqymIyxUx_xWmbdKeDkL5y?T-|4C`&( zqF!oG$!G_al$IIO$5BZJAAqyOvbg0f_@HChT&{yB2_Gm zvy$GO%of1gg4PiMo^Y@_K7hE{WIQy4jS6-uNqQH44 z;22G2#??$br74MQUP-GNbxeyZc{u$P%cj$C7>p^-Oqjhstc_6=I;Es4u*+IXqQ+r8 z3cZlbVpB+3OT!i*@sGlBE9|;Ge1l|{S_;MVmHCDmNJ2n!~k(q;OxYs;wr`+BVP+-RghZSZmw6Zp{c|4Ck3G6J{ z?qR*l+&GG)ht)`Wa4@)e!3NSGIH15uIW4b)Tgen1!l()j8Q)0{DbnG_hbaq#p{Z$@ zzsMYk!6O!o!7)EY&#Otj0F!+1X29BE0>dF8#vK_^Mgq#`vl`2QRJQK^)i(16#L8+kupcE-Tu?%`~EuBZ>j z7M#|Eao)`K7tJG$xl~qPFrIMA6K4O=&4VK&Usks7-90c8*|S?w;521h_s8v1rh#Cr zEVE;jm@7mW&W1r>W$}!dKURR1ls$_64xVY}nKqsYRwQ7W1pQc~$0t!o_*N(+{E9MG zDk(WQj+M=*DHvE~B0zQw49x3XIHMIE7>WcG3>W@vdKcbz)!xJJ%)&0MS0yiF7}+ z+q7In=j1v1m%xYe@YB*fn#Q0P$og%|%RK!i_*c#TIVNv0Pups0Ht#-gyxJM)M*N5b z;;{<0UxJHuM#H(vtq zw+AAWtmP*`p6ay7(+^0H;C`b2_zlD5IX=P0kGGQ=fm?~$$8U6zaIBGd`Vw?J&zm*o z{fA6m#4vB=Jpj%Fmb1EllF8>R0^EPDcOS@O`};Y-(nhv^KgY`0{%hgl^1S_Tg8z=_ rY!<2};i}LKlV`^&=<|n_7I(-;VBG3E@&(xVpM&v1LOJxoYX5%#w1Be= literal 0 HcmV?d00001 diff --git a/network/h_dynamic_cnn_gpu_cpp_v2_pre/test.sh b/network/h_dynamic_cnn_gpu_cpp_v2_pre/test.sh new file mode 100644 index 0000000..0a5f120 --- /dev/null +++ b/network/h_dynamic_cnn_gpu_cpp_v2_pre/test.sh @@ -0,0 +1,14 @@ +make o/kernel_helper_functions.o +make o/kernel_phxy_fill_with_h.o +make o/kernel_phxy_fill_with_spike_selected_w.o +make o/kernel_phxy_one_over_sum_into_pxy.o +make o/kernel_phxy_plus_phxy.o +make o/kernel_phxy_plus_pxy.o +make o/kernel_phxy_times_phxy_equals_phxy.o +make o/kernel_phxy_times_pxy.o +make o/kernel_pxy_plus_v.o +make o/kernel_pxy_reciprocal.o +make o/kernel_pxy_set_to_v.o +make o/kernel_pxy_time_pxy.o +make o/kernel_pxy_times_spike_selected_sxy.o +make o/kernel_pxy_times_v.o