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

/**
 * Frees and clears (sets to NULL) each non-null pointer in the given
 * array. This is likely to be faster than freeing them one-by-one. If
 * footers are used, pointers that have been allocated in different
 * mspaces are not freed or cleared, and the count of all such pointers
 * is returned. For large arrays of pointers with poor locality, it may
 * be worthwhile to sort this array before calling bulk_free.
 */
size_t bulk_free(void *array[], size_t nelem) {
  /*
   * Try to free all pointers in the given array. Note: this could be
   * made faster, by delaying consolidation, at the price of disabling
   * some user integrity checks, We still optimize some consolidations
   * by combining adjacent chunks before freeing, which will occur often
   * if allocated with ialloc or the array is sorted.
   */
  size_t unfreed = 0;
  if (!PREACTION(g_dlmalloc)) {
    void **a;
    void **fence = &(array[nelem]);
    for (a = array; a != fence; ++a) {
      void *mem = *a;
      if (mem != 0) {
        mchunkptr p = mem2chunk(ADDRESS_DEATH_ACTION(mem));
        size_t psize = chunksize(p);
#if FOOTERS
        if (get_mstate_for(p) != g_dlmalloc) {
          ++unfreed;
          continue;
        }
#endif
        check_inuse_chunk(g_dlmalloc, p);
        *a = 0;
        if (RTCHECK(ok_address(g_dlmalloc, p) && ok_inuse(p))) {
          void **b = a + 1; /* try to merge with next chunk */
          mchunkptr next = next_chunk(p);
          if (b != fence && *b == chunk2mem(next)) {
            size_t newsize = chunksize(next) + psize;
            set_inuse(g_dlmalloc, p, newsize);
            *b = chunk2mem(p);
          } else
            dlmalloc_dispose_chunk(g_dlmalloc, p, psize);
        } else {
          CORRUPTION_ERROR_ACTION(g_dlmalloc);
          break;
        }
      }
    }
    if (should_trim(g_dlmalloc, g_dlmalloc->topsize)) {
      dlmalloc_sys_trim(g_dlmalloc, 0);
    }
    POSTACTION(g_dlmalloc);
  }
  return unfreed;
}