
#include "bench_harness.h"
#include "bench_utils.h"
#include <math.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#define UPS 3
#define DOWNS 2
#define NTAPS 8
static const double g_taps[UPS][NTAPS] = {
    {0.02, 0.05, 0.1, 0.25, 0.25, 0.1, 0.05, 0.02},
    {0.01, 0.04, 0.09, 0.22, 0.22, 0.09, 0.04, 0.01},
    {0.005, 0.03, 0.08, 0.2, 0.2, 0.08, 0.03, 0.005}};
static inline void design_polyphase(double bank[UPS][NTAPS]) {
  for (int p = 0; p < UPS; p++) {
    for (int t = 0; t < NTAPS; t++) {
      bank[p][t] = g_taps[p][t];
    }
  }
}
static inline int resample_block(int n, const double *x,
                                 const double bank[UPS][NTAPS], double *y) {
  long long t = 0;
  int outN = 0;
  while ((t / UPS) < n) {
    int phase = (int)(t % UPS);
    int pos = (int)(t / UPS);
    double acc = 0.0;
    for (int k = 0; k < NTAPS; k++) {
      int idx = pos - k;
      if (idx >= 0 && idx < n) {
        acc += x[idx] * bank[phase][k];
      }
    }
    y[outN++] = acc;
    t += DOWNS;
  }
  return outN;
}
static double pipeline_run(int n, const double *inbuf, double *outbuf) {
  double bank[UPS][NTAPS];
  design_polyphase(bank);
  int outN = 0;

  outN = resample_block(n, inbuf, bank, outbuf);

  double acc = 0.0;
  for (int i = 0; i < outN; i++) {
    acc += fabs(outbuf[i]);
  }
  return acc;
}
BENCH_MAIN_SCALAR3(
    T004_Module_011, POLYRES, 4096, 16384, 65536,
    double *inbuf = (double *)malloc((size_t)n * sizeof(double));
    double *outbuf = (double *)malloc((size_t)(n * 2) * sizeof(double));
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      for (int i = 0; i < n; i++) {
        inbuf[i] = bench_rng_double_signed(&rng);
      }
    },
    ans_scalar = pipeline_run(n, inbuf, outbuf), ans_scalar, free(inbuf);
    free(outbuf);)
