
#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, const int *rx1, const int *ry1, const int *rx2,
                const int *ry2, double *ans_out) {
  const int MAXC = 4096;
  int mrect = n / 4;
  if (mrect < 1)
    mrect = 1;
  int E = 2 * mrect;
  int *ex = (int *)malloc((size_t)E * sizeof(int));
  int *ey1 = (int *)malloc((size_t)E * sizeof(int));
  int *ey2 = (int *)malloc((size_t)E * sizeof(int));
  int *edelta = (int *)malloc((size_t)E * sizeof(int));
  int *nxtEvent = (int *)malloc((size_t)E * sizeof(int));
  int *headX = (int *)malloc((size_t)MAXC * sizeof(int));
  int *used = (int *)malloc((size_t)MAXC * sizeof(int));
  int *yvals = (int *)malloc((size_t)MAXC * sizeof(int));
  int *ymap = (int *)malloc((size_t)MAXC * sizeof(int));
  int *el = (int *)malloc((size_t)E * sizeof(int));
  int *er = (int *)malloc((size_t)E * sizeof(int));
  if (!ex || !ey1 || !ey2 || !edelta || !nxtEvent || !headX || !used ||
      !yvals || !ymap || !el || !er) {
    if (ex)
      free(ex);
    if (ey1)
      free(ey1);
    if (ey2)
      free(ey2);
    if (edelta)
      free(edelta);
    if (nxtEvent)
      free(nxtEvent);
    if (headX)
      free(headX);
    if (used)
      free(used);
    if (yvals)
      free(yvals);
    if (ymap)
      free(ymap);
    if (el)
      free(el);
    if (er)
      free(er);
    *ans_out = 0.0;
    return;
  }
  for (int i = 0; i < MAXC; i++) {
    used[i] = 0;
  }
  for (int i = 0; i < mrect; i++) {
    int x1 = rx1[i];
    int y1 = ry1[i];
    int x2 = rx2[i];
    int y2 = ry2[i];
    if (x2 < x1) {
      int tmp = x1;
      x1 = x2;
      x2 = tmp;
    }
    if (y2 < y1) {
      int tmp = y1;
      y1 = y2;
      y2 = tmp;
    }
    if (x1 < 0)
      x1 = 0;
    if (y1 < 0)
      y1 = 0;
    if (x2 >= MAXC)
      x2 = MAXC - 1;
    if (y2 >= MAXC)
      y2 = MAXC - 1;
    if (x2 <= x1)
      x2 = x1 + 1;
    if (y2 <= y1)
      y2 = y1 + 1;
    if (x2 >= MAXC)
      x2 = MAXC - 1;
    if (y2 >= MAXC)
      y2 = MAXC - 1;
    int e0 = 2 * i;
    int e1 = 2 * i + 1;
    ex[e0] = x1;
    ey1[e0] = y1;
    ey2[e0] = y2;
    edelta[e0] = +1;
    ex[e1] = x2;
    ey1[e1] = y1;
    ey2[e1] = y2;
    edelta[e1] = -1;
    used[y1] = 1;
    used[y2] = 1;
  }
  int ny = 0;
  for (int y = 0; y < MAXC; y++) {
    if (used[y]) {
      yvals[ny] = y;
      ymap[y] = ny;
      ny++;
    }
  }
  if (ny < 2) {
    double ans_double = 0.0;

    ans_double = 0.0;

    *ans_out = ans_double;
    free(ex);
    free(ey1);
    free(ey2);
    free(edelta);
    free(nxtEvent);
    free(headX);
    free(used);
    free(yvals);
    free(ymap);
    free(el);
    free(er);
    return;
  }
  int nleaf = ny - 1;
  for (int i = 0; i < E; i++) {
    int yy1 = ey1[i];
    int yy2 = ey2[i];
    int L = ymap[yy1];
    int R = ymap[yy2] - 1;
    if (R < L)
      R = L - 1;
    el[i] = L;
    er[i] = R;
  }
  for (int xx = 0; xx < MAXC; xx++) {
    headX[xx] = -1;
  }
  for (int i = 0; i < E; i++) {
    int xx = ex[i];
    if (xx < 0)
      xx = 0;
    if (xx >= MAXC)
      xx = MAXC - 1;
    nxtEvent[i] = headX[xx];
    headX[xx] = i;
  }
  int segSize = (nleaf > 0) ? (4 * nleaf) : 4;
  int *cover = (int *)malloc((size_t)segSize * sizeof(int));
  int *segLen = (int *)malloc((size_t)segSize * sizeof(int));
  if (!cover || !segLen) {
    if (cover)
      free(cover);
    if (segLen)
      free(segLen);
    free(ex);
    free(ey1);
    free(ey2);
    free(edelta);
    free(nxtEvent);
    free(headX);
    free(used);
    free(yvals);
    free(ymap);
    free(el);
    free(er);
    *ans_out = 0.0;
    return;
  }
  memset(cover, 0, (size_t)segSize * sizeof(int));
  memset(segLen, 0, (size_t)segSize * sizeof(int));
  long long area = 0;
  int prev_x = 0;
  int have_prev = 0;
  for (int xx = 0; xx < MAXC; xx++) {
    if (have_prev) {
      int dx = xx - prev_x;
      if (dx > 0) {
        long long curLen = (nleaf > 0) ? (long long)segLen[1] : 0LL;
        area += curLen * (long long)dx;
      }
    }
    for (int e = headX[xx]; e != -1; e = nxtEvent[e]) {
      int ql = el[e];
      int qr = er[e];
      int delta = edelta[e];
      if (nleaf > 0 && ql <= qr) {
        int stack_node[128];
        int stack_l[128];
        int stack_r[128];
        int stack_state[128];
        int sp = 0;
        stack_node[sp] = 1;
        stack_l[sp] = 0;
        stack_r[sp] = nleaf - 1;
        stack_state[sp] = 0;
        sp++;
        while (sp > 0) {
          sp--;
          int node = stack_node[sp];
          int l_ = stack_l[sp];
          int r_ = stack_r[sp];
          int st = stack_state[sp];
          if (st == 0) {
            if (ql <= l_ && r_ <= qr) {
              cover[node] += delta;
              if (cover[node] > 0) {
                segLen[node] = yvals[r_ + 1] - yvals[l_];
              } else if (l_ == r_) {
                segLen[node] = 0;
              } else {
                segLen[node] = segLen[node * 2] + segLen[node * 2 + 1];
              }
              continue;
            }
            int mid = (l_ + r_) >> 1;
            stack_node[sp] = node;
            stack_l[sp] = l_;
            stack_r[sp] = r_;
            stack_state[sp] = 1;
            sp++;
            if (qr > mid) {
              stack_node[sp] = node * 2 + 1;
              stack_l[sp] = mid + 1;
              stack_r[sp] = r_;
              stack_state[sp] = 0;
              sp++;
            }
            if (ql <= mid) {
              stack_node[sp] = node * 2;
              stack_l[sp] = l_;
              stack_r[sp] = mid;
              stack_state[sp] = 0;
              sp++;
            }
          } else {
            if (cover[node] > 0) {
              segLen[node] = yvals[r_ + 1] - yvals[l_];
            } else if (l_ == r_) {
              segLen[node] = 0;
            } else {
              segLen[node] = segLen[node * 2] + segLen[node * 2 + 1];
            }
          }
        }
      }
    }
    have_prev = 1;
    prev_x = xx;
  }
  double ans_double = 0.0;

  ans_double = (double)area;

  *ans_out = ans_double;
  free(ex);
  free(ey1);
  free(ey2);
  free(edelta);
  free(nxtEvent);
  free(headX);
  free(used);
  free(yvals);
  free(ymap);
  free(el);
  free(er);
  free(cover);
  free(segLen);
}
BENCH_MAIN_SCALAR3(
    T003_Code_047, RECTUNI, 4096, 16384, 65536,
    int *rx1 = (int *)malloc((size_t)n * sizeof(int));
    int *ry1 = (int *)malloc((size_t)n * sizeof(int));
    int *rx2 = (int *)malloc((size_t)n * sizeof(int));
    int *ry2 = (int *)malloc((size_t)n * sizeof(int)); double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      int mrect = n / 4;
      if (mrect < 1)
        mrect = 1;
      for (int i = 0; i < mrect; i++) {
        int bx = (int)(bench_rng_next(&rng) % 2048ULL);
        int by = (int)(bench_rng_next(&rng) % 2048ULL);
        int w = (int)(bench_rng_next(&rng) % 32ULL) + 1;
        int h = (int)(bench_rng_next(&rng) % 32ULL) + 1;
        int x1 = bx;
        int y1 = by;
        int x2 = bx + w;
        int y2 = by + h;
        if (x2 >= 4096)
          x2 = 4095;
        if (y2 >= 4096)
          y2 = 4095;
        if (x2 <= x1)
          x2 = x1 + 1;
        if (y2 <= y1)
          y2 = y1 + 1;
        rx1[i] = x1;
        ry1[i] = y1;
        rx2[i] = x2;
        ry2[i] = y2;
      }
    },
    kernel_run(n, rx1, ry1, rx2, ry2, &ans_scalar), ans_scalar, free(rx1);
    free(ry1); free(rx2); free(ry2);)
