#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 uint64_t *lim_ptr, double *ans_out) {
  (void)n;
  uint64_t LIMIT = *lim_ptr;
  int digits[32];
  int L = 0;
  {
    if (LIMIT == 0ULL) {
      digits[0] = 0;
      L = 1;
    } else {
      uint64_t tmp = LIMIT;
      int tmpdig[32];
      int tlen = 0;
      while (tmp > 0ULL) {
        tmpdig[tlen++] = (int)(tmp % 10ULL);
        tmp /= 10ULL;
      }
      for (int i = 0; i < tlen; i++) {
        digits[i] = tmpdig[tlen - 1 - i];
      }
      L = tlen;
    }
  }
  uint64_t curr[2][3];
  uint64_t next_[2][3];
  for (int t = 0; t < 2; t++)
    for (int m = 0; m < 3; m++)
      curr[t][m] = 0ULL;
  curr[1][0] = 1ULL;
  for (int pos = 0; pos < L; pos++) {
    for (int t = 0; t < 2; t++)
      for (int m = 0; m < 3; m++)
        next_[t][m] = 0ULL;
    for (int tight = 0; tight <= 1; tight++) {
      for (int mod = 0; mod < 3; mod++) {
        uint64_t ways = curr[tight][mod];
        if (ways == 0ULL)
          continue;
        int maxd = tight ? digits[pos] : 9;
        for (int dig = 0; dig <= maxd; dig++) {
          int ntight = (tight && (dig == maxd)) ? 1 : 0;
          int nmod = (mod + dig) % 3;
          next_[ntight][nmod] += ways;
        }
      }
    }
    for (int t = 0; t < 2; t++)
      for (int m = 0; m < 3; m++)
        curr[t][m] = next_[t][m];
  }
  uint64_t good = curr[0][0] + curr[1][0];
  double base_ans = (double)good;
  double ans_double = 0.0;

  ans_double = base_ans;

  *ans_out = ans_double;
}
BENCH_MAIN_SCALAR3(
    T003_Code_033, DIGDP, 4096, 16384, 65536,
    uint64_t *lim = (uint64_t *)malloc(sizeof(uint64_t));
    double ans_scalar = 0.0;
    ,
    {
      bench_rng64_t rng = bench_rng_init(seed);
      *lim = bench_rng_next(&rng) % 1000000000000ULL;
    },
    kernel_run(n, lim, &ans_scalar), ans_scalar, free(lim);)
