#include "bench_harness.h"
#include "bench_utils.h"
#include <stdint.h>
#include <stdlib.h>
static inline long long intersect_count(int u, int v, const int *rowptr,
                                        const int *colind) {
  int iu = rowptr[u];
  int iv = rowptr[v];
  int endu = rowptr[u + 1];
  int endv = rowptr[v + 1];
  long long c = 0;
  while (iu < endu && iv < endv) {
    int a = colind[iu];
    int b = colind[iv];
    if (a <= v) {
      iu++;
      continue;
    }
    if (b <= v) {
      iv++;
      continue;
    }
    if (a == b) {
      c++;
      iu++;
      iv++;
    } else if (a < b) {
      iu++;
    } else {
      iv++;
    }
  }
  return c;
}
static inline void sort_adj_list(int *arr, int len) {
  for (int i = 1; i < len; i++) {
    int key = arr[i];
    int j = i - 1;
    while (j >= 0 && arr[j] > key) {
      arr[j + 1] = arr[j];
      j--;
    }
    arr[j + 1] = key;
  }
}
static double run_triangles(int n, int *rowptr, int *colind) {
  for (int u = 0; u < n; u++) {
    int start = rowptr[u];
    int len = rowptr[u + 1] - start;
    sort_adj_list(colind + start, len);
  }
  long long tri = 0;
  for (int u = 0; u < n; u++) {
    int a = rowptr[u];
    int b = rowptr[u + 1];
    for (int e = a; e < b; e++) {
      int v = colind[e];
      if (v > u) {
        tri += intersect_count(u, v, rowptr, colind);
      }
    }
  }
  double ans = (double)tri;
  double outv = 0.0;
  outv = ans;
  return outv;
}
BENCH_MAIN_SCALAR3(
    T004_Module_048, TRI, 4096, 16384, 65536, int DEG = 4;
    int edges_cap = n * DEG * 2;
    int *temp_u = (int *)malloc((size_t)edges_cap * sizeof(int));
    int *temp_v = (int *)malloc((size_t)edges_cap * sizeof(int));
    int *degcnt = (int *)malloc((size_t)n * sizeof(int));
    int *rowptr = (int *)malloc((size_t)(n + 1) * sizeof(int));
    int *colind = (int *)malloc((size_t)edges_cap * sizeof(int));
    int *cursor = (int *)malloc((size_t)n * sizeof(int));
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      for (int i = 0; i < n; i++)
        degcnt[i] = 0;
      int idx = 0;
      for (int u = 0; u < n; u++) {
        for (int k = 0; k < DEG; k++) {
          int v = (int)(bench_rng_next(&rng) % (unsigned long long)n);
          temp_u[idx] = u;
          temp_v[idx] = v;
          degcnt[u]++;
          idx++;
          temp_u[idx] = v;
          temp_v[idx] = u;
          degcnt[v]++;
          idx++;
        }
      }
      int E = idx;
      int acc = 0;
      for (int i = 0; i < n; i++) {
        rowptr[i] = acc;
        acc += degcnt[i];
      }
      rowptr[n] = acc;
      for (int i = 0; i < n; i++)
        cursor[i] = rowptr[i];
      for (int e = 0; e < E; e++) {
        int u = temp_u[e];
        int v = temp_v[e];
        int p = cursor[u]++;
        colind[p] = v;
      }
    },
    ans_scalar = run_triangles(n, rowptr, colind), ans_scalar, free(temp_u);
    free(temp_v); free(degcnt); free(rowptr); free(colind); free(cursor);)
