
#include "bench_harness.h"
#include "bench_utils.h"
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void kernel_run(int n, int nVar, int mClauses, const int *aVar,
                const unsigned char *aNeg, const int *bVar,
                const unsigned char *bNeg, double *ans_out) {
  double ans_double = 0.0;

  int N = nVar * 2;
  int maxE = mClauses * 2;
  int *head = (int *)malloc((size_t)N * sizeof(int));
  int *to = (int *)malloc((size_t)maxE * sizeof(int));
  int *nxt = (int *)malloc((size_t)maxE * sizeof(int));
  int *rhead = (int *)malloc((size_t)N * sizeof(int));
  int *rto = (int *)malloc((size_t)maxE * sizeof(int));
  int *rnxt = (int *)malloc((size_t)maxE * sizeof(int));
  if (!head || !to || !nxt || !rhead || !rto || !rnxt) {
    if (head)
      free(head);
    if (to)
      free(to);
    if (nxt)
      free(nxt);
    if (rhead)
      free(rhead);
    if (rto)
      free(rto);
    if (rnxt)
      free(rnxt);
    *ans_out = 0.0;
    return;
  }
  for (int i = 0; i < N; i++) {
    head[i] = -1;
    rhead[i] = -1;
  }
  int ecnt = 0;
  for (int i = 0; i < mClauses; i++) {
    int va = aVar[i] % nVar;
    int vb = bVar[i] % nVar;
    if (va < 0)
      va = 0;
    if (vb < 0)
      vb = 0;
    int negA = (aNeg[i] & 1);
    int negB = (bNeg[i] & 1);
    int a_lit = negA ? (2 * va) : (2 * va + 1);
    int b_lit = negB ? (2 * vb) : (2 * vb + 1);
    int a_not = negA ? (2 * va + 1) : (2 * va);
    int b_not = negB ? (2 * vb + 1) : (2 * vb);
    {
      int u = a_not;
      int v = b_lit;
      to[ecnt] = v;
      nxt[ecnt] = head[u];
      head[u] = ecnt;
      rto[ecnt] = u;
      rnxt[ecnt] = rhead[v];
      rhead[v] = ecnt;
      ecnt++;
    }
    {
      int u = b_not;
      int v = a_lit;
      to[ecnt] = v;
      nxt[ecnt] = head[u];
      head[u] = ecnt;
      rto[ecnt] = u;
      rnxt[ecnt] = rhead[v];
      rhead[v] = ecnt;
      ecnt++;
    }
  }
  int *vis = (int *)calloc((size_t)N, sizeof(int));
  int *order = (int *)malloc((size_t)N * sizeof(int));
  int *stack = (int *)malloc((size_t)N * sizeof(int));
  int *edge_it = (int *)malloc((size_t)N * sizeof(int));
  if (!vis || !order || !stack || !edge_it) {
    if (vis)
      free(vis);
    if (order)
      free(order);
    if (stack)
      free(stack);
    if (edge_it)
      free(edge_it);
    free(head);
    free(to);
    free(nxt);
    free(rhead);
    free(rto);
    free(rnxt);
    *ans_out = 0.0;
    return;
  }
  int order_sz = 0;
  for (int start = 0; start < N; start++) {
    if (vis[start])
      continue;
    int sp = 0;
    stack[sp] = start;
    edge_it[sp] = head[start];
    vis[start] = 1;
    while (sp >= 0) {
      int u = stack[sp];
      int e = edge_it[sp];
      if (e != -1) {
        int v = to[e];
        edge_it[sp] = nxt[e];
        if (!vis[v]) {
          vis[v] = 1;
          sp++;
          stack[sp] = v;
          edge_it[sp] = head[v];
        }
      } else {
        order[order_sz++] = u;
        sp--;
      }
    }
  }
  int *comp = (int *)malloc((size_t)N * sizeof(int));
  if (!comp) {
    free(vis);
    free(order);
    free(stack);
    free(edge_it);
    free(head);
    free(to);
    free(nxt);
    free(rhead);
    free(rto);
    free(rnxt);
    *ans_out = 0.0;
    return;
  }
  for (int i = 0; i < N; i++)
    comp[i] = -1;
  int comp_id = 0;
  for (int idx = order_sz - 1; idx >= 0; idx--) {
    int v0 = order[idx];
    if (comp[v0] != -1)
      continue;
    int sp = 0;
    stack[sp] = v0;
    edge_it[sp] = rhead[v0];
    comp[v0] = comp_id;
    while (sp >= 0) {
      int u = stack[sp];
      int e = edge_it[sp];
      if (e != -1) {
        int w = rto[e];
        edge_it[sp] = rnxt[e];
        if (comp[w] == -1) {
          comp[w] = comp_id;
          sp++;
          stack[sp] = w;
          edge_it[sp] = rhead[w];
        }
      } else {
        sp--;
      }
    }
    comp_id++;
  }
  int sat = 1;
  for (int v = 0; v < nVar; v++) {
    if (comp[2 * v] == comp[2 * v + 1]) {
      sat = 0;
      break;
    }
  }
  if (!sat) {
    ans_double = -1.0;
  } else {
    int cnt_true = 0;
    for (int v = 0; v < nVar; v++) {
      if (comp[2 * v] > comp[2 * v + 1])
        cnt_true++;
    }
    ans_double = (double)cnt_true;
  }
  free(vis);
  free(order);
  free(stack);
  free(edge_it);
  free(comp);
  free(head);
  free(to);
  free(nxt);
  free(rhead);
  free(rto);
  free(rnxt);

  *ans_out = ans_double;
}
BENCH_MAIN_SCALAR3(
    T003_Code_018, TWOSAT, 4096, 16384, 65536, int nVar = n / 512 + 4;
    if (nVar > 256) nVar = 256; int mClauses = nVar * 4;
    int *aVar = (int *)malloc((size_t)mClauses * sizeof(int));
    unsigned char *aNeg = (unsigned char *)malloc((size_t)mClauses *
                                                  sizeof(unsigned char));
    int *bVar = (int *)malloc((size_t)mClauses * sizeof(int));
    unsigned char *bNeg = (unsigned char *)malloc((size_t)mClauses *
                                                  sizeof(unsigned char));
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      for (int i = 0; i < mClauses; i++) {
        aVar[i] = (int)(bench_rng_next(&rng) % (uint64_t)nVar);
        bVar[i] = (int)(bench_rng_next(&rng) % (uint64_t)nVar);
        aNeg[i] = (unsigned char)(bench_rng_next(&rng) & 1ULL);
        bNeg[i] = (unsigned char)(bench_rng_next(&rng) & 1ULL);
      }
    },
    kernel_run(n, nVar, mClauses, aVar, aNeg, bVar, bNeg, &ans_scalar),
    ans_scalar, free(aVar);
    free(aNeg); free(bVar); free(bNeg);)
