#include "bench_harness.h"
#include "bench_utils.h"
#include <math.h>
#include <stdint.h>
#include <stdlib.h>
#define RADIUS 2
static void integral_image(const uint8_t *img, int w, int h, int32_t *ii) {
  int sw = w + 1;
  for (int y = 0; y <= h; y++) {
    for (int x = 0; x <= w; x++) {
      ii[y * sw + x] = 0;
    }
  }
  for (int y = 0; y < h; y++) {
    int rowsum = 0;
    for (int x = 0; x < w; x++) {
      rowsum += (int)img[y * w + x];
      ii[(y + 1) * sw + (x + 1)] = ii[y * sw + (x + 1)] + rowsum;
    }
  }
}
static inline int clampi_int2(int v, int lo, int hi) {
  if (v < lo)
    return lo;
  if (v > hi)
    return hi;
  return v;
}
static double boxfilter_sum(const int32_t *ii, int w, int h) {
  int sw = w + 1;
  double acc = 0.0;
  for (int y = 0; y < h; y++) {
    for (int x = 0; x < w; x++) {
      int x1 = clampi_int2(x - RADIUS, 0, w - 1);
      int y1 = clampi_int2(y - RADIUS, 0, h - 1);
      int x2 = clampi_int2(x + RADIUS, 0, w - 1);
      int y2 = clampi_int2(y + RADIUS, 0, h - 1);
      int A = ii[y1 * sw + x1];
      int B = ii[y1 * sw + (x2 + 1)];
      int C = ii[(y2 + 1) * sw + x1];
      int D = ii[(y2 + 1) * sw + (x2 + 1)];
      int sum = D - B - C + A;
      int area = (x2 - x1 + 1) * (y2 - y1 + 1);
      acc += (double)sum / (double)area;
    }
  }
  return acc;
}
static double pipeline_run(int w, int h, const uint8_t *img) {
  int32_t *ii =
      (int32_t *)malloc((size_t)((w + 1) * (h + 1)) * sizeof(int32_t));
  integral_image(img, w, h, ii);
  double val = boxfilter_sum(ii, w, h);
  free(ii);
  double ans = 0.0;

  ans = val;
  return ans;
}
BENCH_MAIN_SCALAR3(
    T004_Module_004, IMG04, 4096, 16384, 65536,
    int w = (int)(sqrt((double)n) + 0.5);
    int h = w; uint8_t *img = (uint8_t *)malloc((size_t)(n) * sizeof(uint8_t));
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      for (int i = 0; i < n; i++)
        img[i] = (uint8_t)(bench_rng_next(&rng) & 255ULL);
    },
    ans_scalar = pipeline_run(w, h, img), ans_scalar, free(img);)
