#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 int hexval(char c) {
  if (c >= '0' && c <= '9')
    return (int)(c - '0');
  if (c >= 'A' && c <= 'F')
    return (int)(c - 'A' + 10);
  if (c >= 'a' && c <= 'f')
    return (int)(c - 'a' + 10);
  return 0;
}
static int pct_decode_inplace(char *s, int len) {
  int w = 0;
  int i = 0;
  while (i < len) {
    char c = s[i];
    if (c == '%' && i + 2 < len) {
      int hi = hexval(s[i + 1]);
      int lo = hexval(s[i + 2]);
      unsigned char v = (unsigned char)((hi << 4) | lo);
      s[w++] = (char)v;
      i += 3;
    } else {
      s[w++] = c;
      i++;
    }
  }
  return w;
}
static void parse_url_line(const char *line, int len, uint64_t *h) {
  int i = 0;
  while (i < len && line[i] != ':')
    i++;
  i += 3;
  if (i > len)
    return;
  int host_start = i;
  while (i < len && line[i] != '/' && line[i] != '\n' && line[i] != '\r')
    i++;
  int host_end = i;
  int path_start = i;
  while (i < len && line[i] != '\n' && line[i] != '\r')
    i++;
  int path_end = i;
  char hostbuf[128];
  int hostlen = host_end - host_start;
  if (hostlen > 127)
    hostlen = 127;
  if (hostlen < 0)
    hostlen = 0;
  memcpy(hostbuf, line + host_start, (size_t)hostlen);
  hostbuf[hostlen] = 0;
  char pathbuf[256];
  int rawlen = path_end - path_start;
  if (rawlen > 255)
    rawlen = 255;
  if (rawlen < 0)
    rawlen = 0;
  memcpy(pathbuf, line + path_start, (size_t)rawlen);
  int dec_len = pct_decode_inplace(pathbuf, rawlen);
  *h = hash_bytes64(*h, hostbuf, hostlen);
  *h = hash_bytes64(*h, pathbuf, dec_len);
}
static double pipeline_run(const char *urls, int url_len) {
  uint64_t total = 0;

  uint64_t h = 1469598103934665603ULL;
  int pos = 0;
  while (pos < url_len) {
    int start = pos;
    while (pos < url_len && urls[pos] != '\n')
      pos++;
    int l = pos - start;
    if (l > 0)
      parse_url_line(urls + start, l, &h);
    if (pos < url_len && urls[pos] == '\n')
      pos++;
  }

  return (double)total;
}
BENCH_MAIN_SCALAR3(
    T004_Module_023, URLP, 4096, 16384, 65536,
    char *urls = (char *)malloc((size_t)n * 64);
    int url_len = 0; double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      url_len = 0;
      for (int i = 0; i < n; i++) {
        unsigned long long r = (unsigned long long)bench_rng_next(&rng);
        const char *scheme = (i & 1) ? "http" : "https";
        char tmp[128];
        int tlen = snprintf(tmp, sizeof(tmp),
                            "%s://example%u.com/p/%02llx%%2F%02llx/%02llx\n",
                            scheme, (unsigned)(i % 1000), r & 0xffULL,
                            (r >> 8) & 0xffULL, (r >> 16) & 0xffULL);
        if (url_len + tlen < n * 64) {
          memcpy(urls + url_len, tmp, (size_t)tlen);
          url_len += tlen;
        } else {
          break;
        }
      }
    },
    ans_scalar = pipeline_run(urls, url_len);
    , ans_scalar, free(urls);)
