// bench_harness.h - thin, macro-only main boilerplate you can reuse
#ifndef BENCH_HARNESS_H
#define BENCH_HARNESS_H

#include "bench_utils.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

/* Common input scales (three cases) and outer-loop levels (1..3).
 * The repetitive main() logic is provided by macros; you only need to
 * supply ALLOC / INIT / CALL / FREE and one line to compute the checksum.
 *
 * Three common wrappers are provided:
 *   1) double scalar result (tolerant hash)
 *   2) double array result (tolerant hash)
 *   3) non‑tolerant byte/integer array result (e.g., fixed-point/bit ops)
 */

#define BENCH_MAIN_SCALAR3(FNAME, KID, N1, N2, N3, ALLOC, INIT, CALL, SCALAR,  \
                           FREE)                                               \
  int main(int argc, char **argv) {                                            \
    int case_id = (argc > 1) ? atoi(argv[1]) : 1;                              \
    int level = (argc > 2) ? atoi(argv[2]) : 1;                                \
    if (case_id < 1 || case_id > 3)                                            \
      case_id = 1;                                                             \
    if (level < 0 || level > 8)                                                \
      level = 1;                                                               \
    int n;                                                                     \
    long long loop;                                                            \
    switch (case_id) {                                                         \
    case 1:                                                                    \
      n = (N1);                                                                \
      break;                                                                   \
    case 2:                                                                    \
      n = (N2);                                                                \
      break;                                                                   \
    default:                                                                   \
    case 3:                                                                    \
      n = (N3);                                                                \
      break;                                                                   \
    }                                                                          \
    loop = bench_outer_loops(level);                                           \
    uint64_t seed = bench_seed(case_id);                                       \
    ALLOC;                                                                     \
    INIT;                                                                      \
    bench_timer_start();                                                       \
    for (long long __bh_l = 0; __bh_l < loop; ++__bh_l) {                      \
      CALL;                                                                    \
    }                                                                          \
    double secs = bench_timer_stop_seconds();                                  \
    bench_tol_t tol = bench_default_tol();                                     \
    uint64_t hash = bench_tol_hash_double((double)(SCALAR), tol);              \
    printf(#FNAME ",K" #KID                                                    \
                  ",case%d,L%d,checksum=0x%016llx,time=%.6f,loops=%lld\n",     \
           case_id, level, (unsigned long long)hash, secs, loop);              \
    FREE;                                                                      \
    return 0;                                                                  \
  }

/* 2) double array result (tolerant hash) */
#define BENCH_MAIN_ARRAY3_D(FNAME, KID, N1, N2, N3, ALLOC, INIT, CALL, HPTR,   \
                            HN, FREE)                                          \
  int main(int argc, char **argv) {                                            \
    int case_id = (argc > 1) ? atoi(argv[1]) : 1;                              \
    int level = (argc > 2) ? atoi(argv[2]) : 1;                                \
    if (case_id < 1 || case_id > 3)                                            \
      case_id = 1;                                                             \
    if (level < 0 || level > 8)                                                \
      level = 1;                                                               \
    int n;                                                                     \
    long long loop;                                                            \
    switch (case_id) {                                                         \
    case 1:                                                                    \
      n = (N1);                                                                \
      break;                                                                   \
    case 2:                                                                    \
      n = (N2);                                                                \
      break;                                                                   \
    default:                                                                   \
    case 3:                                                                    \
      n = (N3);                                                                \
      break;                                                                   \
    }                                                                          \
    loop = bench_outer_loops(level);                                           \
    uint64_t seed = bench_seed(case_id);                                       \
    ALLOC;                                                                     \
    INIT;                                                                      \
    bench_timer_start();                                                       \
    for (long long __bh_l = 0; __bh_l < loop; ++__bh_l) {                      \
      CALL;                                                                    \
    }                                                                          \
    double secs = bench_timer_stop_seconds();                                  \
    bench_tol_t tol = bench_default_tol();                                     \
    uint64_t hash =                                                            \
        bench_tol_hash_arrayd((const double *)(HPTR), (size_t)(HN), tol);      \
    printf(#FNAME ",K" #KID                                                    \
                  ",case%d,L%d,checksum=0x%016llx,time=%.6f,loops=%lld\n",     \
           case_id, level, (unsigned long long)hash, secs, loop);              \
    FREE;                                                                      \
    return 0;                                                                  \
  }

/* 3) non‑tolerant (e.g., fixed-point / integer arrays) */
#define BENCH_MAIN_ARRAY3_BYTES(FNAME, KID, N1, N2, N3, ALLOC, INIT, CALL,     \
                                HPTR, NBYTES, FREE)                            \
  int main(int argc, char **argv) {                                            \
    int case_id = (argc > 1) ? atoi(argv[1]) : 1;                              \
    int level = (argc > 2) ? atoi(argv[2]) : 1;                                \
    if (case_id < 1 || case_id > 3)                                            \
      case_id = 1;                                                             \
    if (level < 0 || level > 8)                                                \
      level = 1;                                                               \
    int n;                                                                     \
    long long loop;                                                            \
    switch (case_id) {                                                         \
    case 1:                                                                    \
      n = (N1);                                                                \
      break;                                                                   \
    case 2:                                                                    \
      n = (N2);                                                                \
      break;                                                                   \
    default:                                                                   \
    case 3:                                                                    \
      n = (N3);                                                                \
      break;                                                                   \
    }                                                                          \
    loop = bench_outer_loops(level);                                           \
    uint64_t seed = bench_seed(case_id);                                       \
    ALLOC;                                                                     \
    INIT;                                                                      \
    bench_timer_start();                                                       \
    for (long long __bh_l = 0; __bh_l < loop; ++__bh_l) {                      \
      CALL;                                                                    \
    }                                                                          \
    double secs = bench_timer_stop_seconds();                                  \
    uint64_t hash = bench_hash_bytes((const void *)(HPTR), (size_t)(NBYTES));  \
    printf(#FNAME ",K" #KID                                                    \
                  ",case%d,L%d,checksum=0x%016llx,time=%.6f,loops=%lld\n",     \
           case_id, level, (unsigned long long)hash, secs, loop);              \
    FREE;                                                                      \
    return 0;                                                                  \
  }

#endif /* BENCH_HARNESS_H */
