#include "bench_harness.h"
#include "bench_utils.h"
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
typedef struct {
  int level;
  int user;
  double val;
} LogRec;
#define LSEED 0x9e3779b9u
static inline uint32_t mix32_l(uint32_t x) {
  x ^= LSEED;
  x ^= x >> 16;
  x *= 0x7feb352dU;
  x ^= x >> 15;
  x *= 0x846ca68bU;
  x ^= x >> 16;
  return x;
}
static inline int filt_level_ge3(const LogRec *r) { return r->level >= 3; }
static inline void proj_user_val(const LogRec *r, int *uid, double *v) {
  *uid = r->user;
  *v = r->val;
}
static size_t next_pow2_l(size_t x) {
  size_t p = 1;
  while (p < x)
    p <<= 1;
  return p;
}
static double pipeline_run(int n, const LogRec *recs) {
  size_t need = (size_t)((n * 10) / 7 + 1);
  size_t cap = next_pow2_l(need);
  int *keys = (int *)malloc(cap * sizeof(int));
  double *aggs = (double *)malloc(cap * sizeof(double));
  unsigned char *used = (unsigned char *)malloc(cap * sizeof(unsigned char));
  if (!keys || !aggs || !used) {
    if (keys)
      free(keys);
    if (aggs)
      free(aggs);
    if (used)
      free(used);
    return 0.0;
  }
  memset(used, 0, cap * sizeof(unsigned char));
  for (int i = 0; i < n; i++) {
    const LogRec *r = &recs[i];
    if (!filt_level_ge3(r))
      continue;
    int uid;
    double v;
    proj_user_val(r, &uid, &v);
    size_t idx = (size_t)(mix32_l((uint32_t)uid) & (uint32_t)(cap - 1));
    while (used[idx] && keys[idx] != uid)
      idx = (idx + 1) & (cap - 1);
    if (used[idx]) {
      aggs[idx] += v;
    } else {
      used[idx] = 1;
      keys[idx] = uid;
      aggs[idx] = v;
    }
  }
  double total = 0.0;
  for (size_t i = 0; i < cap; i++) {
    if (used[i])
      total += aggs[i];
  }
  free(keys);
  free(aggs);
  free(used);
  double ans_once = total;
  double ans = 0.0;

  ans = ans_once;
  return ans;
}
BENCH_MAIN_SCALAR3(
    T004_Module_060, LOGPIPE, 4096, 16384, 65536,
    LogRec *recs = (LogRec *)malloc((size_t)n * sizeof(LogRec));
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      int userN = 1024;
      for (int i = 0; i < n; i++) {
        recs[i].level = (int)(bench_rng_next(&rng) % 6ULL);
        recs[i].user = (int)(bench_rng_next(&rng) % (unsigned long long)userN);
        recs[i].val = bench_rng_double_signed(&rng);
      }
    },
    ans_scalar = pipeline_run(n, recs), ans_scalar, free(recs);)
