
#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 *edge_u, const int *edge_v,
                const int *edge_cap, double *ans_out) {
  double ans_double = 0.0;

  int s = 0;
  int t = V - 1;
  if (t < 0)
    t = 0;
  int maxe = E * 2 + 2;
  int *head = (int *)malloc((size_t)V * sizeof(int));
  int *to = (int *)malloc((size_t)maxe * sizeof(int));
  int *cap = (int *)malloc((size_t)maxe * sizeof(int));
  int *nxt = (int *)malloc((size_t)maxe * sizeof(int));
  int *rev = (int *)malloc((size_t)maxe * sizeof(int));
  if (!head || !to || !cap || !nxt || !rev) {
    if (head)
      free(head);
    if (to)
      free(to);
    if (cap)
      free(cap);
    if (nxt)
      free(nxt);
    if (rev)
      free(rev);
    *ans_out = 0.0;
    return;
  }
  for (int i = 0; i < V; i++)
    head[i] = -1;
  int ecnt = 0;
  for (int i = 0; i < E; i++) {
    int u = edge_u[i];
    int v = edge_v[i];
    int c = edge_cap[i];
    if (u < 0)
      u = 0;
    if (u >= V)
      u %= V;
    if (v < 0)
      v = 0;
    if (v >= V)
      v %= V;
    if (c < 0)
      c = -c;
    if (c == 0)
      c = 1;
    to[ecnt] = v;
    cap[ecnt] = c;
    nxt[ecnt] = head[u];
    rev[ecnt] = ecnt + 1;
    head[u] = ecnt;
    to[ecnt + 1] = u;
    cap[ecnt + 1] = 0;
    nxt[ecnt + 1] = head[v];
    rev[ecnt + 1] = ecnt;
    head[v] = ecnt + 1;
    ecnt += 2;
  }
  long long maxflow = 0;
  int *parent_edge = (int *)malloc((size_t)V * sizeof(int));
  int *q = (int *)malloc((size_t)V * sizeof(int));
  if (!parent_edge || !q) {
    if (parent_edge)
      free(parent_edge);
    if (q)
      free(q);
    free(head);
    free(to);
    free(cap);
    free(nxt);
    free(rev);
    *ans_out = 0.0;
    return;
  }
  while (1) {
    for (int i = 0; i < V; i++)
      parent_edge[i] = -1;
    parent_edge[s] = -2;
    int qh = 0, qt = 0;
    q[qt++] = s;
    while (qh < qt && parent_edge[t] == -1) {
      int u = q[qh++];
      for (int e = head[u]; e != -1; e = nxt[e]) {
        if (cap[e] > 0) {
          int v = to[e];
          if (parent_edge[v] == -1) {
            parent_edge[v] = e;
            if (v == t)
              break;
            q[qt++] = v;
          }
        }
      }
    }
    if (parent_edge[t] == -1)
      break;
    int aug = 2147483647;
    int v = t;
    while (v != s) {
      int e = parent_edge[v];
      if (cap[e] < aug)
        aug = cap[e];
      v = to[rev[e]];
    }
    v = t;
    while (v != s) {
      int e = parent_edge[v];
      cap[e] -= aug;
      cap[rev[e]] += aug;
      v = to[rev[e]];
    }
    maxflow += (long long)aug;
  }
  ans_double = (double)maxflow;
  free(parent_edge);
  free(q);
  free(head);
  free(to);
  free(cap);
  free(nxt);
  free(rev);

  *ans_out = ans_double;
}
BENCH_MAIN_SCALAR3(
    T003_Code_015, DINIC, 4096, 16384, 65536, int V = n / 1024 + 4;
    if (V > 64) V = 64; int baseE = (V > 1 ? (V - 1) : 0); int extraE = V * 3;
    int E = baseE + extraE; int *eu = (int *)malloc((size_t)E * sizeof(int));
    int *ev = (int *)malloc((size_t)E * sizeof(int));
    int *ec = (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;
        ec[idx] = (int)((bench_rng_next(&rng) % 20ULL) + 1ULL);
        idx++;
      }
      while (idx < E) {
        int u = (int)(bench_rng_next(&rng) % (uint64_t)V);
        int v = (int)(bench_rng_next(&rng) % (uint64_t)V);
        if (u == v)
          v = (v + 1) % V;
        eu[idx] = u;
        ev[idx] = v;
        ec[idx] = (int)((bench_rng_next(&rng) % 20ULL) + 1ULL);
        idx++;
      }
    },
    kernel_run(n, V, E, eu, ev, ec, &ans_scalar), ans_scalar, free(eu);
    free(ev); free(ec);)
