#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;
}
static inline char lower_char(char c) {
  if (c >= 'A' && c <= 'Z')
    return (char)(c + ('a' - 'A'));
  return c;
}
static void lower_inplace(char *s, int len) {
  for (int i = 0; i < len; i++)
    s[i] = lower_char(s[i]);
}
static uint64_t parse_headers_block(const char *buf, int len) {
  struct HeaderSlot {
    int nlen;
    int vlen;
    char name[32];
    char value[256];
  };
  struct HeaderSlot slots[16];
  int slot_count = 0;
  int i = 0;
  while (i < len) {
    if (i + 1 < len && buf[i] == '\r' && buf[i + 1] == '\n') {
      i += 2;
      if (i + 1 < len && buf[i] == '\r' && buf[i + 1] == '\n') {
        i += 2;
        break;
      }
      continue;
    }
    int name_start = i;
    while (i < len && buf[i] != ':' && buf[i] != '\r' && buf[i] != '\n')
      i++;
    int name_end = i;
    while (i < len && buf[i] != ':') {
      if (buf[i] == '\r' || buf[i] == '\n')
        break;
      i++;
    }
    if (i >= len || buf[i] != ':') {
      while (i < len && !(buf[i] == '\r' && i + 1 < len && buf[i + 1] == '\n'))
        i++;
      if (i + 1 < len)
        i += 2;
      continue;
    }
    i++;
    while (i < len && (buf[i] == ' ' || buf[i] == '\t'))
      i++;
    int value_start = i;
    while (i < len && !(buf[i] == '\r' && i + 1 < len && buf[i + 1] == '\n'))
      i++;
    int value_end = i;
    if (i + 1 < len)
      i += 2;
    int nlen = name_end - name_start;
    if (nlen > 31)
      nlen = 31;
    if (nlen < 0)
      nlen = 0;
    char namebuf[32];
    memcpy(namebuf, buf + name_start, (size_t)nlen);
    namebuf[nlen] = 0;
    lower_inplace(namebuf, nlen);
    int vlen = value_end - value_start;
    if (vlen > 255)
      vlen = 255;
    if (vlen < 0)
      vlen = 0;
    char valbuf[256];
    memcpy(valbuf, buf + value_start, (size_t)vlen);
    valbuf[vlen] = 0;
    int idx_slot = -1;
    for (int s = 0; s < slot_count; s++) {
      if (slots[s].nlen == nlen &&
          memcmp(slots[s].name, namebuf, (size_t)nlen) == 0) {
        idx_slot = s;
        break;
      }
    }
    if (idx_slot < 0) {
      if (slot_count < 16) {
        idx_slot = slot_count++;
        slots[idx_slot].nlen = nlen;
        memcpy(slots[idx_slot].name, namebuf, (size_t)nlen);
        slots[idx_slot].name[nlen] = 0;
        slots[idx_slot].vlen = 0;
      } else {
        continue;
      }
    }
    if (slots[idx_slot].vlen < 255) {
      if (slots[idx_slot].vlen > 0 && slots[idx_slot].vlen < 255) {
        slots[idx_slot].value[slots[idx_slot].vlen++] = ',';
      }
      int copylen = vlen;
      if (slots[idx_slot].vlen + copylen > 255) {
        copylen = 255 - slots[idx_slot].vlen;
      }
      memcpy(slots[idx_slot].value + slots[idx_slot].vlen, valbuf,
             (size_t)copylen);
      slots[idx_slot].vlen += copylen;
      slots[idx_slot].value[slots[idx_slot].vlen] = 0;
    }
  }
  uint64_t h = 1469598103934665603ULL;
  for (int s = 0; s < slot_count; s++) {
    h ^= (uint64_t)slots[s].nlen;
    h *= 1099511628211ULL;
    h = hash_bytes64(h, slots[s].name, slots[s].nlen);
    h = hash_bytes64(h, slots[s].value, slots[s].vlen);
  }
  return h;
}
static double pipeline_run(const char *hdr, int hdr_len) {
  uint64_t total = 0;

  uint64_t h = parse_headers_block(hdr, hdr_len);

  return (double)total;
}
BENCH_MAIN_SCALAR3(
    T004_Module_024, HTTPHDR, 4096, 16384, 65536,
    char *hdr = (char *)malloc((size_t)n * 64 + 16);
    int hdr_len = 0; double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      hdr_len = 0;
      for (int i = 0; i < n; i++) {
        const char *name;
        if (i % 4 == 0)
          name = "Host";
        else if (i % 4 == 1)
          name = "User-Agent";
        else if (i % 4 == 2)
          name = "X-Test";
        else
          name = "X-Test";
        unsigned long long v = (unsigned long long)bench_rng_next(&rng);
        char tmp[128];
        int tlen = snprintf(tmp, sizeof(tmp), "%s: val-%04llx\r\n", name,
                            v & 0xffffULL);
        if (hdr_len + tlen + 4 < n * 64 + 16) {
          memcpy(hdr + hdr_len, tmp, (size_t)tlen);
          hdr_len += tlen;
        } else {
          break;
        }
      }
      memcpy(hdr + hdr_len, "\r\n", 2);
      hdr_len += 2;
    },
    ans_scalar = pipeline_run(hdr, hdr_len);
    , ans_scalar, free(hdr);)
