#include "bench_harness.h"
#include "bench_utils.h"
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MF_HASH_SIZE 65536
#define MF_MIN_MATCH 3
#define MF_MAX_PROBE 1
static inline uint32_t mf_hash3(const uint8_t *p) {
  uint32_t h = (uint32_t)p[0] * 16777619u;
  h ^= (uint32_t)p[1];
  h *= 16777619u;
  h ^= (uint32_t)p[2];
  h *= 16777619u;
  return h & (MF_HASH_SIZE - 1);
}
static inline int mf_match_len(const uint8_t *buf, int n, int a, int b) {
  int len = 0;
  while (a + len < n && b + len < n && buf[a + len] == buf[b + len] &&
         len < 255)
    len++;
  return len;
}
static uint64_t mf_process(const uint8_t *in, int n) {
  int *last = (int *)malloc(sizeof(int) * MF_HASH_SIZE);
  for (int i = 0; i < MF_HASH_SIZE; i++)
    last[i] = -1;
  uint64_t acc = 0;
  for (int i = 0; i + 2 < n; i++) {
    uint32_t h = mf_hash3(in + i);
    int bestLen = 0;
    int pos = last[h];
    int tries = 0;
    while (pos >= 0 && tries < MF_MAX_PROBE) {
      int candLen = mf_match_len(in, n, pos, i);
      if (candLen > bestLen)
        bestLen = candLen;
      tries++;
      break;
    }
    if (bestLen >= MF_MIN_MATCH)
      acc += (uint64_t)bestLen;
    last[h] = i;
  }
  free(last);
  return acc;
}
static double pipeline_run(int n, const uint8_t *inbuf) {
  uint64_t score = mf_process(inbuf, n);
  double v = 0.0;

  v = (double)(score & 0xFFFFFFFFFFFFull);
  return v;
}
BENCH_MAIN_SCALAR3(
    T004_Module_038, LZMAMF, 4096, 16384, 65536,
    uint8_t *inbuf = (uint8_t *)malloc((size_t)n);
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      for (int i = 0; i < n; i++)
        inbuf[i] = (uint8_t)(bench_rng_next(&rng) & 0xFFu);
    },
    ans_scalar = pipeline_run(n, inbuf), ans_scalar, free(inbuf);)
