From d18eec3544854e31d9f4b74125eb2e64f1973e6a Mon Sep 17 00:00:00 2001 From: Matteo Perotti Date: Mon, 20 May 2024 15:42:52 +0200 Subject: [PATCH] sw/fdotp: unroll loop to eliminate WAR --- sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.c | 80 +++++++++++++++++++++- sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.h | 8 ++- sw/spatzBenchmarks/dp-fdotp/main.c | 2 +- 3 files changed, 83 insertions(+), 7 deletions(-) diff --git a/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.c b/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.c index d597fa64..762c22cb 100644 --- a/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.c +++ b/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.c @@ -19,7 +19,7 @@ #include "fdotp.h" // 64-bit dot-product: a * b -double fdotp_v64b(const double *a, const double *b, unsigned int avl) { +double fdotp_v64b_m8(const double *a, const double *b, unsigned int avl) { const unsigned int orig_avl = avl; unsigned int vl; @@ -57,8 +57,82 @@ double fdotp_v64b(const double *a, const double *b, unsigned int avl) { return red; } +// 64-bit dot-product: a * b +// m8 allows only for partial register re-allocation with factor-2 unrolling +double fdotp_v64b_m8_unrl(const double *a, const double *b, unsigned int avl) { + const unsigned int orig_avl = avl; + unsigned int vl; + + double red; + + // Stripmine and accumulate a partial reduced vector + do { + // Set the vl + asm volatile("vsetvli %0, %1, e64, m8, ta, ma" : "=r"(vl) : "r"(avl)); + + // Load chunk a and b + asm volatile("vle64.v v8, (%0)" ::"r"(a)); + asm volatile("vle64.v v16, (%0)" ::"r"(b)); + + // Multiply and accumulate + if (avl == orig_avl) { + asm volatile("vfmul.vv v24, v8, v16"); + } else { + asm volatile("vfmacc.vv v24, v8, v16"); + } + + // Bump pointers + a += vl; + b += vl; + avl -= vl; + + if (avl <= 0) break; + + // Set the vl + asm volatile("vsetvli %0, %1, e64, m8, ta, ma" : "=r"(vl) : "r"(avl)); + + // Load chunk a and b + asm volatile("vle64.v v0, (%0)" ::"r"(a)); + asm volatile("vle64.v v8, (%0)" ::"r"(b)); + + // Multiply and accumulate + asm volatile("vfmacc.vv v24, v0, v8"); + + // Bump pointers + a += vl; + b += vl; + avl -= vl; + + if (avl <= 0) break; + + // Set the vl + asm volatile("vsetvli %0, %1, e64, m8, ta, ma" : "=r"(vl) : "r"(avl)); + + // Load chunk a and b + asm volatile("vle64.v v16, (%0)" ::"r"(a)); + asm volatile("vle64.v v0, (%0)" ::"r"(b)); + + // Multiply and accumulate + asm volatile("vfmacc.vv v24, v0, v16"); + + // Bump pointers + a += vl; + b += vl; + avl -= vl; + } while (avl > 0); + + // Clean the accumulator + asm volatile("vmv.s.x v0, zero"); + + // Reduce and return + asm volatile("vfredusum.vs v0, v24, v0"); + asm volatile("vfmv.f.s %0, v0" : "=f"(red)); + + return red; +} + // 32-bit dot-product: a * b -float fdotp_v32b(const float *a, const float *b, unsigned int avl) { +float fdotp_v32b_m8(const float *a, const float *b, unsigned int avl) { const unsigned int orig_avl = avl; unsigned int vl; @@ -93,7 +167,7 @@ float fdotp_v32b(const float *a, const float *b, unsigned int avl) { } // 16-bit dot-product: a * b -_Float16 fdotp_v16b(const _Float16 *a, const _Float16 *b, unsigned int avl) { +_Float16 fdotp_v16b_m8(const _Float16 *a, const _Float16 *b, unsigned int avl) { const unsigned int orig_avl = avl; unsigned int vl; diff --git a/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.h b/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.h index 4b5ac88d..ed928614 100644 --- a/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.h +++ b/sw/spatzBenchmarks/dp-fdotp/kernel/fdotp.h @@ -19,11 +19,13 @@ #ifndef _FDOTPROD_H_ #define _FDOTPROD_H_ -inline double fdotp_v64b(const double *a, const double *b, unsigned int avl) +inline double fdotp_v64b_m8(const double *a, const double *b, unsigned int avl) __attribute__((always_inline)); -inline float fdotp_v32b(const float *a, const float *b, unsigned int avl) +inline double fdotp_v64b_m8_unrl(const double *a, const double *b, unsigned int avl) __attribute__((always_inline)); -inline _Float16 fdotp_v16b(const _Float16 *a, const _Float16 *b, +inline float fdotp_v32b_m8(const float *a, const float *b, unsigned int avl) + __attribute__((always_inline)); +inline _Float16 fdotp_v16b_m8(const _Float16 *a, const _Float16 *b, unsigned int avl) __attribute__((always_inline)); #endif diff --git a/sw/spatzBenchmarks/dp-fdotp/main.c b/sw/spatzBenchmarks/dp-fdotp/main.c index 974bb782..1d291718 100644 --- a/sw/spatzBenchmarks/dp-fdotp/main.c +++ b/sw/spatzBenchmarks/dp-fdotp/main.c @@ -81,7 +81,7 @@ int main() { // Calculate dotp double acc; - acc = fdotp_v64b(a_int, b_int, dim); + acc = fdotp_v64b_m8_unrl(a_int, b_int, dim); result[cid] = acc; // Wait for all cores to finish