#include "bench_harness.h"
#include "bench_utils.h"
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#define MAX_STATES 512
#define MAX_STACK 512
#define ALPHABET_SZ 4
typedef struct {
  int s;
  int e;
} Frag;
typedef struct {
  int out1;
  int out2;
  int ch;
} State;
typedef struct {
  State st[MAX_STATES];
  int nstate;
  int start;
  int accept;
} NFA;
static int alpha_map(char c) {
  if (c == 'a')
    return 0;
  if (c == 'b')
    return 1;
  if (c == 'c')
    return 2;
  if (c == 'd')
    return 3;
  return -1;
}
static int new_state(NFA *nfa, int ch, int o1, int o2) {
  int id = nfa->nstate++;
  nfa->st[id].ch = ch;
  nfa->st[id].out1 = o1;
  nfa->st[id].out2 = o2;
  return id;
}
static void patch(NFA *nfa, int s, int target) {
  if (nfa->st[s].out1 < 0)
    nfa->st[s].out1 = target;
  else
    nfa->st[s].out2 = target;
}
static void append_frags(int *list, int *n, int s) { list[(*n)++] = s; }
static void do_patch_list(NFA *nfa, const int *list, int n, int t) {
  for (int i = 0; i < n; i++)
    patch(nfa, list[i], t);
}
static int precedence(char c) {
  if (c == '*')
    return 3;
  if (c == '.')
    return 2;
  if (c == '|')
    return 1;
  return 0;
}
static int is_alpha(char c) {
  return (c == 'a' || c == 'b' || c == 'c' || c == 'd');
}
static int re_to_post(const char *re, char *out, int outcap) {
  char opstack[MAX_STACK];
  int sp = 0;
  int o = 0;
  int prev_is_atom = 0;
  for (int i = 0; re[i]; i++) {
    char c = re[i];
    if (is_alpha(c)) {
      if (prev_is_atom) {
        while (sp > 0 && precedence(opstack[sp - 1]) >= precedence('.')) {
          if (o < outcap)
            out[o++] = opstack[--sp];
        }
        opstack[sp++] = '.';
      }
      if (o < outcap)
        out[o++] = c;
      prev_is_atom = 1;
    } else if (c == '(') {
      if (prev_is_atom) {
        while (sp > 0 && precedence(opstack[sp - 1]) >= precedence('.')) {
          if (o < outcap)
            out[o++] = opstack[--sp];
        }
        opstack[sp++] = '.';
      }
      opstack[sp++] = '(';
      prev_is_atom = 0;
    } else if (c == ')') {
      while (sp > 0 && opstack[sp - 1] != '(') {
        if (o < outcap)
          out[o++] = opstack[--sp];
      }
      if (sp > 0 && opstack[sp - 1] == '(')
        sp--;
      prev_is_atom = 1;
    } else if (c == '|') {
      while (sp > 0 && precedence(opstack[sp - 1]) >= precedence('|')) {
        if (o < outcap)
          out[o++] = opstack[--sp];
      }
      opstack[sp++] = '|';
      prev_is_atom = 0;
    } else if (c == '*') {
      if (o < outcap)
        out[o++] = c;
      prev_is_atom = 1;
    }
  }
  while (sp > 0) {
    if (o < outcap)
      out[o++] = opstack[--sp];
  }
  if (o < outcap)
    out[o] = 0;
  return o;
}
static void nfa_from_post(const char *post, NFA *nfa) {
  Frag stack[MAX_STACK];
  int sp = 0;
  nfa->nstate = 0;
  for (int i = 0; post[i]; i++) {
    char c = post[i];
    if (is_alpha(c)) {
      int s = new_state(nfa, alpha_map(c), -1, -1);
      int e = new_state(nfa, -1, -1, -1);
      patch(nfa, s, e);
      Frag f = {s, e};
      stack[sp++] = f;
    } else if (c == '.') {
      Frag f2 = stack[--sp];
      Frag f1 = stack[--sp];
      patch(nfa, f1.e, f2.s);
      Frag f = {f1.s, f2.e};
      stack[sp++] = f;
    } else if (c == '|') {
      Frag f2 = stack[--sp];
      Frag f1 = stack[--sp];
      int s = new_state(nfa, -2, f1.s, f2.s);
      int e = new_state(nfa, -1, -1, -1);
      patch(nfa, f1.e, e);
      patch(nfa, f2.e, e);
      Frag f = {s, e};
      stack[sp++] = f;
    } else if (c == '*') {
      Frag f1 = stack[--sp];
      int s = new_state(nfa, -2, f1.s, -1);
      int e = new_state(nfa, -1, -1, -1);
      patch(nfa, f1.e, e);
      patch(nfa, f1.e, s);
      patch(nfa, s, e);
      Frag f = {s, e};
      stack[sp++] = f;
    }
  }
  Frag f = stack[--sp];
  nfa->start = f.s;
  nfa->accept = f.e;
}
static inline void addstate_closure(const NFA *nfa, int s, int *set, int *ns) {
  int stk[MAX_STATES];
  int sp = 0;
  stk[sp++] = s;
  while (sp > 0) {
    int st = stk[--sp];
    int seen = 0;
    for (int i = 0; i < *ns; i++) {
      if (set[i] == st) {
        seen = 1;
        break;
      }
    }
    if (seen)
      continue;
    set[(*ns)++] = st;
    if (nfa->st[st].ch == -2) {
      if (nfa->st[st].out1 >= 0)
        stk[sp++] = nfa->st[st].out1;
      if (nfa->st[st].out2 >= 0)
        stk[sp++] = nfa->st[st].out2;
    }
  }
}
static inline void step(const NFA *nfa, const int *cur, int ncur, int c,
                        int *nxt, int *nnxt) {
  *nnxt = 0;
  for (int i = 0; i < ncur; i++) {
    int st = cur[i];
    int ch = nfa->st[st].ch;
    if (ch >= 0 && ch == c) {
      int o1 = nfa->st[st].out1;
      if (o1 >= 0)
        addstate_closure(nfa, o1, nxt, nnxt);
    }
  }
}
static int nfa_match(const NFA *nfa, const char *s, int len) {
  int cur[MAX_STATES], ncur = 0;
  int nxt[MAX_STATES], nnxt = 0;
  addstate_closure(nfa, nfa->start, cur, &ncur);
  for (int i = 0; i < len; i++) {
    int c = alpha_map(s[i]);
    if (c < 0)
      c = -999;
    nnxt = 0;
    step(nfa, cur, ncur, c, nxt, &nnxt);
    ncur = 0;
    for (int j = 0; j < nnxt; j++)
      cur[j] = nxt[j];
    ncur = nnxt;
  }
  for (int i = 0; i < ncur; i++) {
    if (cur[i] == nfa->accept)
      return 1;
  }
  return 0;
}
static double pipeline_run(const char *patterns, int pcnt, const char *texts,
                           int textlen) {
  double outv = 0.0;

  uint64_t acc = 1469598103934665603ULL;
  int ppos = 0;
  for (int pid = 0; pid < pcnt; pid++) {
    char rebuf[64];
    int rlen = 0;
    while (ppos < pcnt * 8 && patterns[ppos] && patterns[ppos] != ';') {
      if (rlen < 63)
        rebuf[rlen++] = patterns[ppos];
      ppos++;
    }
    if (rlen < 63)
      rebuf[rlen] = 0;
    else
      rebuf[63] = 0;
    ppos++;
    char post[128];
    re_to_post(rebuf, post, 128);
    NFA nfa;
    nfa_from_post(post, &nfa);
    int pos = 0;
    while (pos < textlen) {
      int end = pos;
      while (end < textlen && texts[end] != '\n')
        end++;
      int l = end - pos;
      int m = nfa_match(&nfa, texts + pos, l);
      acc ^= (uint64_t)m + 0x9e3779b97f4a7c15ULL * (uint64_t)pid;
      if (end < textlen && texts[end] == '\n')
        end++;
      pos = end;
    }
  }
  outv += (double)acc;

  return outv;
}
BENCH_MAIN_SCALAR3(T004_Module_026, REGNFA, 4096, 16384, 65536,
                   char *patterns = (char *)malloc((size_t)n * 8 + 8);
                   char *texts = (char *)malloc((size_t)n * 16 + 16);
                   int pcnt = 0; int textlen = 0; double ans_scalar = 0.0;
                   , ({
                     bench_rng64_t rng = bench_rng_init(seed);
                     pcnt = 0;
                     int w = 0;
                     for (int i = 0; i < n && pcnt < 16; i += 4) {
                       char p[16];
                       int t = (int)(bench_rng_next(&rng) % 3ULL);
                       if (t == 0)
                         snprintf(p, sizeof(p), "a*b;");
                       else if (t == 1)
                         snprintf(p, sizeof(p), "(ab|c)*;");
                       else
                         snprintf(p, sizeof(p), "a(cd|b)*;");
                       int plen = (int)strlen(p);
                       if (pcnt * 8 + plen < (int)(n * 8 + 8)) {
                         memcpy(patterns + pcnt * 8, p, plen);
                         if (plen < 8)
                           memset(patterns + pcnt * 8 + plen, 0, 8 - plen);
                         pcnt++;
                       }
                     }
                     textlen = 0;
                     for (int i = 0; i < n; i++) {
                       unsigned long long r = bench_rng_next(&rng);
                       char line[32];
                       int llen = 0;
                       for (int k = 0; k < 8; k++) {
                         char tbl[5] = {'a', 'b', 'c', 'd', 'a'};
                         line[llen++] = tbl[(r >> (2 * k)) & 3ULL];
                       }
                       line[llen++] = '\n';
                       if (textlen + llen < (int)(n * 16 + 16)) {
                         memcpy(texts + textlen, line, llen);
                         textlen += llen;
                       } else
                         break;
                     }
                   }),
                   ans_scalar = pipeline_run(patterns, pcnt, texts, textlen),
                   ans_scalar, free(patterns);
                   free(texts);)
