#include "libc/mem/mem.h"
#include "third_party/dlmalloc/dlmalloc.internal.h"

/**
 * Returns (by copy) a struct containing various summary statistics:
 *
 * - arena: current total non-mmapped bytes allocated from system
 *
 * - ordblks: the number of free chunks
 *
 * - smblks: always zero.
 *
 * - hblks: current number of mmapped regions
 *
 * - hblkhd: total bytes held in mmapped regions
 *
 * - usmblks: the maximum total allocated space. This will be greater
 *   than current total if trimming has occurred.
 *
 * - fsmblks: always zero
 *
 * - uordblks: current total allocated space (normal or mmapped)
 *
 * - fordblks: total free space
 *
 * - keepcost: the maximum number of bytes that could ideally be
 *   released back to system via malloc_trim. ("ideally" means that it
 *   ignores page restrictions etc.)
 *
 * Because these fields are ints, but internal bookkeeping may
 * be kept as longs, the reported values may wrap around zero and
 * thus be inaccurate.
 */
struct mallinfo mallinfo(void) {
  struct mallinfo nm = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
  ensure_initialization();
  if (!PREACTION(g_dlmalloc)) {
    check_malloc_state(g_dlmalloc);
    if (is_initialized(g_dlmalloc)) {
      size_t nfree = SIZE_T_ONE; /* top always free */
      size_t mfree = g_dlmalloc->topsize + TOP_FOOT_SIZE;
      size_t sum = mfree;
      msegmentptr s = &g_dlmalloc->seg;
      while (s != 0) {
        mchunkptr q = align_as_chunk(s->base);
        while (segment_holds(s, q) && q != g_dlmalloc->top &&
               q->head != FENCEPOST_HEAD) {
          size_t sz = chunksize(q);
          sum += sz;
          if (!is_inuse(q)) {
            mfree += sz;
            ++nfree;
          }
          q = next_chunk(q);
        }
        s = s->next;
      }
      nm.arena = sum;
      nm.ordblks = nfree;
      nm.hblkhd = g_dlmalloc->footprint - sum;
      nm.usmblks = g_dlmalloc->max_footprint;
      nm.uordblks = g_dlmalloc->footprint - mfree;
      nm.fordblks = mfree;
      nm.keepcost = g_dlmalloc->topsize;
    }
    POSTACTION(g_dlmalloc);
  }
  return nm;
}