/* finaltest.c: LARGE-SCALE FINALIZATION TEST
 *
 * $Id: //info.ravenbrook.com/project/mps/master/code/finaltest.c#30 $
 * Copyright (c) 2001-2020 Ravenbrook Limited.  See end of file for license.
 * Portions copyright (C) 2002 Global Graphics Software.
 *
 * DESIGN
 *
 * .mode: This test has two modes.
 *
 * .mode.park: In this mode, we use the arena's default generation
 * chain, leave the arena parked and call mps_arena_collect. This
 * tests that the default generation chain works and that all segments
 * get condemned via TraceStartCollectAll. (See job003771 item 4.)
 *
 * .mode.poll: In this mode, we use our own generation chain (with
 * small generations), allocate into generation 1, unclamp the arena,
 * and provoke collection by allocating. This tests that custom
 * generation chains work, and that segments get condemned via
 * TracePoll even if there is no allocation into generation 0 of the
 * chain. (See job003771 item 5.)
 *
 * DEPENDENCIES
 *
 * This test uses the dylan object format, but the reliance on this
 * particular format is not great and could be removed.
 *
 * NOTES
 *
 * This code was created by first copying <code/finalcv.c>
 */

#include "mpm.h"
#include "testlib.h"
#include "mpslib.h"
#include "mps.h"
#include "mpscamc.h"
#include "mpscams.h"
#include "mpscawl.h"
#include "mpsclo.h"
#include "mpsavm.h"
#include "fmtdy.h"
#include "fmtdytst.h"
#include "mpstd.h"

#include <math.h> /* HUGE_VAL */
#include <stdio.h> /* fflush, printf, stdout */

enum {
  ModePARK,                     /* .mode.park */
  ModePOLL                      /* .mode.poll */
};


#define testArenaSIZE   ((size_t)16<<20)
#define rootCOUNT 20
#define maxtreeDEPTH 9
#define collectionCOUNT 10


/* global object counter */

static mps_word_t object_count = 0;

static mps_word_t make_numbered_cons(mps_word_t car, mps_word_t cdr,
                                     mps_ap_t ap)
{
    mps_word_t cons;
    die(make_dylan_vector(&cons, ap, 3), "make_dylan_vector");
    DYLAN_VECTOR_SLOT(cons, 0) = car;
    DYLAN_VECTOR_SLOT(cons, 1) = cdr;
    DYLAN_VECTOR_SLOT(cons, 2) = DYLAN_INT(object_count);
    ++ object_count;
    return cons;
}

static mps_word_t make_numbered_tree(mps_word_t depth,
                                     mps_ap_t ap)
{
    mps_word_t left, right;
    if (depth < 2) {
        left = DYLAN_INT(object_count);
        right = DYLAN_INT(object_count);
    } else {
        left = make_numbered_tree(depth-1, ap);
        right = make_numbered_tree(depth-1, ap);
    }
    return make_numbered_cons(left, right, ap);
}

static void register_numbered_tree(mps_word_t tree, mps_arena_t arena)
{
    /* don't finalize ints */
    if ((tree & 1) == 0) {
        mps_addr_t tree_ref = (mps_addr_t)tree;
        die(mps_finalize(arena, &tree_ref), "mps_finalize");
        register_numbered_tree(DYLAN_VECTOR_SLOT(tree, 0), arena);
        register_numbered_tree(DYLAN_VECTOR_SLOT(tree, 1), arena);
    }
}

static mps_word_t make_indirect_cons(mps_word_t car, mps_word_t cdr,
                                     mps_ap_t ap)
{
    mps_word_t cons, indirect;
    die(make_dylan_vector(&indirect, ap, 1), "make_dylan_vector");
    DYLAN_VECTOR_SLOT(indirect, 0) = DYLAN_INT(object_count);
    die(make_dylan_vector(&cons, ap, 3), "make_dylan_vector");
    DYLAN_VECTOR_SLOT(cons, 0) = car;
    DYLAN_VECTOR_SLOT(cons, 1) = cdr;
    DYLAN_VECTOR_SLOT(cons, 2) = indirect;
    ++ object_count;
    return cons;
}

static mps_word_t make_indirect_tree(mps_word_t depth,
                                     mps_ap_t ap)
{
    mps_word_t left, right;
    if (depth < 2) {
        left = DYLAN_INT(object_count);
        right = DYLAN_INT(object_count);
    } else {
        left = make_indirect_tree(depth-1, ap);
        right = make_indirect_tree(depth-1, ap);
    }
    return make_indirect_cons(left, right, ap);
}

static void register_indirect_tree(mps_word_t tree, mps_arena_t arena)
{
    /* don't finalize ints */
    if ((tree & 1) == 0) {
        mps_word_t indirect = DYLAN_VECTOR_SLOT(tree,2);
        mps_addr_t indirect_ref = (mps_addr_t)indirect;
        die(mps_finalize(arena, &indirect_ref), "mps_finalize");
        register_indirect_tree(DYLAN_VECTOR_SLOT(tree, 0), arena);
        register_indirect_tree(DYLAN_VECTOR_SLOT(tree, 1), arena);
    }
}

static void *root[rootCOUNT];

static void test_trees(int mode, const char *name, mps_arena_t arena,
                       mps_pool_t pool, mps_ap_t ap,
                       mps_word_t (*make)(mps_word_t, mps_ap_t),
                       void (*reg)(mps_word_t, mps_arena_t))
{
  size_t collections = 0;
  size_t finals = 0;
  size_t i;
  int object_alloc;
  PoolClass klass = ClassOfPoly(Pool, pool);

  object_count = 0;

  printf("---- Mode %s, pool class %s, %s trees ----\n",
         mode == ModePARK ? "PARK" : "POLL",
         ClassName(klass), name);
  mps_arena_park(arena);
  mps_message_type_enable(arena, mps_message_type_gc());

  /* make some trees */
  for(i = 0; i < rootCOUNT; ++i) {
    root[i] = (void *)(*make)(maxtreeDEPTH, ap);
    (*reg)((mps_word_t)root[i], arena);
  }

  /* clean out the roots */
  for(i = 0; i < rootCOUNT; ++i) {
    root[i] = 0;
  }

  while (finals < object_count && collections < collectionCOUNT) {
    mps_message_type_t type;
    mps_word_t final_this_time = 0;
    switch (mode) {
    default:
    case ModePARK:
      printf("Collecting...");
      (void)fflush(stdout);
      die(mps_arena_collect(arena), "collect");
      printf(" Done.\n");
      break;
    case ModePOLL:
      mps_arena_release(arena);
      printf("Allocating...");
      (void)fflush(stdout);
      object_alloc = 0;
      while (object_alloc < 1000 && !mps_message_poll(arena))
        (void)DYLAN_INT(object_alloc++);
      printf(" Done.\n");
      break;
    }
    {
      size_t live_size = (object_count - finals) * sizeof(void *) * 3;
      size_t total_size = mps_pool_total_size(pool);
      size_t free_size = mps_pool_free_size(pool);
      Insist(free_size <= total_size);
      Insist(free_size + live_size <= total_size);
    }
    while (mps_message_queue_type(&type, arena)) {
      mps_message_t message;
      cdie(mps_message_get(&message, arena, type), "message_get");
      if (type == mps_message_type_finalization()) {
        mps_addr_t objaddr;
        mps_message_finalization_ref(&objaddr, arena, message);
        ++ final_this_time;
      } else if (type == mps_message_type_gc()) {
        ++ collections;
      } else {
        error("Unexpected message type %lu.", (unsigned long)type);
      }
      mps_message_discard(arena, message);
    }
    finals += final_this_time;
    printf("%"PRIuLONGEST" objects finalized: total %"PRIuLONGEST
           " of %"PRIuLONGEST"\n", (ulongest_t)final_this_time,
           (ulongest_t)finals, (ulongest_t)object_count);
  }

  if (finals != object_count) {
    PoolClass poolClass = ClassOfPoly(Pool, BufferOfAP(ap)->pool);
    error("Not all objects were finalized for %s in mode %s.",
          ClassName(poolClass),
          mode == ModePOLL ? "POLL" : "PARK");
  }

  if (collections > collectionCOUNT)
    error("Expected no more than %lu collections but got %lu.",
          (unsigned long)collectionCOUNT, (unsigned long)collections);
}

static void test_pool(int mode, mps_arena_t arena, mps_chain_t chain,
                      mps_pool_class_t pool_class)
{
  mps_ap_t ap;
  mps_fmt_t fmt;
  mps_pool_t pool;
  mps_root_t mps_root;

  die(mps_fmt_create_A(&fmt, arena, dylan_fmt_A()), "fmt_create\n");
  MPS_ARGS_BEGIN(args) {
    MPS_ARGS_ADD(args, MPS_KEY_FORMAT, fmt);
    if (mode == ModePOLL) {
      MPS_ARGS_ADD(args, MPS_KEY_CHAIN, chain);
      MPS_ARGS_ADD(args, MPS_KEY_GEN, 1);
    }
    die(mps_pool_create_k(&pool, arena, pool_class, args),
        "pool_create\n");
  } MPS_ARGS_END(args);
  die(mps_root_create_table(&mps_root, arena, mps_rank_exact(), (mps_rm_t)0,
                            root, (size_t)rootCOUNT),
      "root_create\n");
  die(mps_ap_create(&ap, pool, mps_rank_exact()), "ap_create\n");

  test_trees(mode, "numbered", arena, pool, ap, make_numbered_tree,
             register_numbered_tree);
  test_trees(mode, "indirect", arena, pool, ap, make_indirect_tree,
             register_indirect_tree);

  mps_ap_destroy(ap);
  mps_root_destroy(mps_root);
  mps_pool_destroy(pool);
  mps_fmt_destroy(fmt);
}


static void test_mode(int mode, mps_arena_t arena, mps_chain_t chain)
{
  test_pool(mode, arena, chain, mps_class_amc());
  test_pool(mode, arena, chain, mps_class_amcz());
  test_pool(mode, arena, chain, mps_class_ams());
  test_pool(mode, arena, chain, mps_class_awl());
  test_pool(mode, arena, chain, mps_class_lo());
}


int main(int argc, char *argv[])
{
  mps_arena_t arena;
  mps_thr_t thread;
  mps_chain_t chain;
  mps_gen_param_s params[2];
  size_t gens = 2;
  size_t i;

  testlib_init(argc, argv);

  MPS_ARGS_BEGIN(args) {
    /* Randomize pause time as a regression test for job004007. */
    double t = rnd_double();
    if (t == 0.0)
      t = HUGE_VAL; /* Would prefer to use INFINITY but it's not in C89. */
    else
      t = 1 / t - 1;
    MPS_ARGS_ADD(args, MPS_KEY_PAUSE_TIME, t);
    MPS_ARGS_ADD(args, MPS_KEY_ARENA_SIZE, testArenaSIZE);
    die(mps_arena_create_k(&arena, mps_arena_class_vm(), args),
        "arena_create\n");
  } MPS_ARGS_END(args);
  mps_message_type_enable(arena, mps_message_type_finalization());
  die(mps_thread_reg(&thread, arena), "thread_reg\n");
  for (i = 0; i < gens; ++i) {
    params[i].mps_capacity = 1;
    params[i].mps_mortality = 0.5;
  }
  die(mps_chain_create(&chain, arena, gens, params), "chain_create\n");

  test_mode(ModePOLL, arena, chain);
  test_mode(ModePARK, arena, NULL);

  mps_arena_park(arena);
  mps_chain_destroy(chain);
  mps_thread_dereg(thread);
  mps_arena_destroy(arena);

  printf("%s: Conclusion: Failed to find any defects.\n", argv[0]);
  return 0;
}


/* C. COPYRIGHT AND LICENSE
 *
 * Copyright (C) 2001-2020 Ravenbrook Limited <https://www.ravenbrook.com/>.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are
 * met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the
 *    distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */