#include "bench_harness.h"
#include "bench_utils.h"
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
static inline uint64_t hash_bytes64(uint64_t h, const char *s, int len) {
  for (int i = 0; i < len; i++) {
    h ^= (uint64_t)(unsigned char)s[i];
    h *= 1099511628211ULL;
  }
  return h;
}
typedef struct {
  uint64_t acc;
} SaxCtx;
typedef struct {
  void (*on_open)(void *, const char *, int);
  void (*on_close)(void *, const char *, int);
  void (*on_text)(void *, const char *, int);
} SaxHandler;
static void cb_open(void *ud, const char *name, int len) {
  SaxCtx *ctx = (SaxCtx *)ud;
  ctx->acc = hash_bytes64(ctx->acc, name, len);
}
static void cb_close(void *ud, const char *name, int len) {
  SaxCtx *ctx = (SaxCtx *)ud;
  ctx->acc = hash_bytes64(ctx->acc, name, len);
}
static void cb_text(void *ud, const char *txt, int len) {
  SaxCtx *ctx = (SaxCtx *)ud;
  ctx->acc = hash_bytes64(ctx->acc, txt, len);
}
static void sax_parse(const char *buf, int len, const SaxHandler *h,
                      void *user) {
  int i = 0;
  while (i < len) {
    if (buf[i] == '<') {
      if (i + 1 < len && buf[i + 1] == '/') {
        int j = i + 2;
        while (j < len) {
          char c = buf[j];
          if (c == '>' || c == ' ' || c == '\t' || c == '\n' || c == '\r')
            break;
          j++;
        }
        if (h->on_close && j > i + 2)
          h->on_close(user, buf + i + 2, j - (i + 2));
        while (j < len && buf[j] != '>')
          j++;
        if (j < len)
          j++;
        i = j;
      } else {
        int j = i + 1;
        while (j < len) {
          char c = buf[j];
          if (c == '>' || c == ' ' || c == '\t' || c == '\n' || c == '\r')
            break;
          j++;
        }
        if (h->on_open && j > i + 1)
          h->on_open(user, buf + i + 1, j - (i + 1));
        while (j < len && buf[j] != '>')
          j++;
        if (j < len)
          j++;
        i = j;
      }
    } else {
      int j = i;
      while (j < len && buf[j] != '<')
        j++;
      if (h->on_text && j > i)
        h->on_text(user, buf + i, j - i);
      i = j;
    }
  }
}
static double pipeline_run(const char *buf, int buflen) {
  SaxHandler handler;
  handler.on_open = cb_open;
  handler.on_close = cb_close;
  handler.on_text = cb_text;

  SaxCtx ctx;
  ctx.acc = 1469598103934665603ULL;
  sax_parse(buf, buflen, &handler, &ctx);

  return (double)ctx.acc;
}

BENCH_MAIN_SCALAR3(
    T004_Module_021, XMLSAX, 4096, 16384, 65536,
    char *xmlbuf = (char *)malloc((size_t)n * 32);
    int buflen = 0; double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      buflen = 0;
      for (int i = 0; i < n; i++) {
        const char *tag = (i % 3 == 0) ? "aaa" : ((i % 3 == 1) ? "bbb" : "ccc");
        unsigned long long v = (unsigned long long)bench_rng_next(&rng);
        char tmp[64];
        int tlen = snprintf(tmp, sizeof(tmp), "<%s>%08llx</%s>", tag,
                            v & 0xffffffffULL, tag);
        if (buflen + tlen < n * 32) {
          memcpy(xmlbuf + buflen, tmp, (size_t)tlen);
          buflen += tlen;
        } else {
          break;
        }
      }
    },
    ans_scalar += pipeline_run(xmlbuf, buflen);
    , ans_scalar, free(xmlbuf);)
