
#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 V, int E, const int *eu_in, const int *ev_in,
                double *ans_out) {
  double ans_double = 0.0;

  int *eu = (int *)malloc((size_t)E * sizeof(int));
  int *ev = (int *)malloc((size_t)E * sizeof(int));
  if (!eu || !ev) {
    if (eu)
      free(eu);
    if (ev)
      free(ev);
    *ans_out = 0.0;
    return;
  }
  for (int i = 0; i < E; i++) {
    int a = eu_in[i];
    int b = ev_in[i];
    if (a < 0)
      a = 0;
    if (a >= V)
      a %= V;
    if (b < 0)
      b = 0;
    if (b >= V)
      b %= V;
    eu[i] = a;
    ev[i] = b;
  }
  int *visited = (int *)malloc((size_t)V * sizeof(int));
  int *stack = (int *)malloc((size_t)V * sizeof(int));
  unsigned char *removed_node = (unsigned char *)calloc((size_t)V, 1);
  unsigned char *removed_edge = (unsigned char *)calloc((size_t)E, 1);
  unsigned char *is_bridge = (unsigned char *)calloc((size_t)E, 1);
  if (!visited || !stack || !removed_node || !removed_edge || !is_bridge) {
    if (visited)
      free(visited);
    if (stack)
      free(stack);
    if (removed_node)
      free(removed_node);
    if (removed_edge)
      free(removed_edge);
    if (is_bridge)
      free(is_bridge);
    free(eu);
    free(ev);
    *ans_out = 0.0;
    return;
  }
  int num_art = 0;
  for (int rem = 0; rem < V; rem++) {
    for (int i = 0; i < V; i++) {
      visited[i] = 0;
    }
    for (int i = 0; i < V; i++) {
      removed_node[i] = 0;
    }
    removed_node[rem] = 1;
    int comps = 0;
    for (int start = 0; start < V; start++) {
      if (removed_node[start])
        continue;
      if (visited[start])
        continue;
      comps++;
      int sp = 0;
      stack[sp++] = start;
      visited[start] = 1;
      while (sp > 0) {
        int u = stack[--sp];
        for (int e = 0; e < E; e++) {
          if (removed_edge[e])
            continue;
          int a = eu[e];
          int b = ev[e];
          if (removed_node[a] || removed_node[b])
            continue;
          int w = -1;
          if (a == u)
            w = b;
          else if (b == u)
            w = a;
          if (w != -1 && !visited[w]) {
            visited[w] = 1;
            stack[sp++] = w;
          }
        }
      }
    }
    if (comps > 1)
      num_art++;
  }
  int num_bridge = 0;
  for (int e_rm = 0; e_rm < E; e_rm++) {
    for (int i = 0; i < V; i++) {
      visited[i] = 0;
    }
    for (int i = 0; i < E; i++) {
      removed_edge[i] = 0;
    }
    removed_edge[e_rm] = 1;
    for (int i = 0; i < V; i++) {
      removed_node[i] = 0;
    }
    int comps = 0;
    for (int start = 0; start < V; start++) {
      if (visited[start])
        continue;
      comps++;
      int sp = 0;
      stack[sp++] = start;
      visited[start] = 1;
      while (sp > 0) {
        int u = stack[--sp];
        for (int e = 0; e < E; e++) {
          if (removed_edge[e])
            continue;
          int a = eu[e];
          int b = ev[e];
          int w = -1;
          if (a == u)
            w = b;
          else if (b == u)
            w = a;
          if (w != -1 && !visited[w]) {
            visited[w] = 1;
            stack[sp++] = w;
          }
        }
      }
    }
    if (comps > 1) {
      num_bridge++;
      is_bridge[e_rm] = 1;
    }
  }
  for (int i = 0; i < V; i++) {
    visited[i] = 0;
  }
  for (int i = 0; i < E; i++) {
    removed_edge[i] = is_bridge[i];
  }
  for (int i = 0; i < V; i++) {
    removed_node[i] = 0;
  }
  int comps_bcc = 0;
  for (int start = 0; start < V; start++) {
    if (visited[start])
      continue;
    comps_bcc++;
    int sp = 0;
    stack[sp++] = start;
    visited[start] = 1;
    while (sp > 0) {
      int u = stack[--sp];
      for (int e = 0; e < E; e++) {
        if (removed_edge[e])
          continue;
        int a = eu[e];
        int b = ev[e];
        int w = -1;
        if (a == u)
          w = b;
        else if (b == u)
          w = a;
        if (w != -1 && !visited[w]) {
          visited[w] = 1;
          stack[sp++] = w;
        }
      }
    }
  }
  ans_double = (double)(num_art + num_bridge + comps_bcc);
  free(visited);
  free(stack);
  free(removed_node);
  free(removed_edge);
  free(is_bridge);
  free(eu);
  free(ev);

  *ans_out = ans_double;
}
BENCH_MAIN_SCALAR3(
    T003_Code_020, BRIDGE, 4096, 16384, 65536, int V = n / 1024 + 4;
    if (V > 64) V = 64; int base_chain = (V > 1 ? (V - 1) : 0);
    int extraE = V * 2; int E = base_chain + extraE;
    int *eu = (int *)malloc((size_t)E * sizeof(int));
    int *ev = (int *)malloc((size_t)E * sizeof(int)); double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      int idx = 0;
      for (int i = 0; i < V - 1 && idx < E; i++) {
        eu[idx] = i;
        ev[idx] = i + 1;
        idx++;
      }
      while (idx < E) {
        int a = (int)(bench_rng_next(&rng) % (uint64_t)V);
        int b = (int)(bench_rng_next(&rng) % (uint64_t)V);
        if (a == b)
          b = (b + 1) % V;
        eu[idx] = a;
        ev[idx] = b;
        idx++;
      }
    },
    kernel_run(n, V, E, eu, ev, &ans_scalar), ans_scalar, free(eu);
    free(ev);)
