/*
 * Copyright (c) 1997
 * Mark of the Unicorn, Inc.
 *
 * Permission to use, copy, modify, distribute and sell this software
 * and its documentation for any purpose is hereby granted without fee,
 * provided that the above copyright notice appear in all copies and
 * that both that copyright notice and this permission notice appear
 * in supporting documentation.  Mark of the Unicorn makes no
 * representations about the suitability of this software for any
 * purpose.  It is provided "as is" without express or implied warranty.
 */
/***********************************************************************************
  LeakCheck.h

    SUMMARY: A suite of template functions for verifying the behavior of
      operations in the presence of exceptions. Requires that the operations
      be written so that each operation that could cause an exception causes
      simulate_possible_failure() to be called (see "nc_alloc.h").

***********************************************************************************/
#ifndef INCLUDED_MOTU_LeakCheck
#define INCLUDED_MOTU_LeakCheck 1

#include "Prefix.h"

#include "nc_alloc.h"

#include <cstdio>
#include <cassert>
#include <iterator>

#include <iostream>

EH_BEGIN_NAMESPACE

template <class T1, class T2>
inline ostream& operator << (
ostream& s,
const pair <T1, T2>& p) {
    return s<<'['<<p.first<<":"<<p.second<<']';
}
EH_END_NAMESPACE

/*===================================================================================
  CheckInvariant

  EFFECTS:  Generalized function to check an invariant on a container. Specialize
    this for particular containers if such a check is available.
====================================================================================*/
template <class C>
void CheckInvariant(const C&)
{}

/*===================================================================================
  WeakCheck

  EFFECTS: Given a value and an operation, repeatedly applies the operation to a
    copy of the value triggering the nth possible exception, where n increments
    with each repetition until no exception is thrown or max_iters is reached.
    Reports any detected memory leaks and checks any invariant defined for the
    value type whether the operation succeeds or fails.
====================================================================================*/
template <class Value, class Operation>
void WeakCheck(const Value& v, const Operation& op, long max_iters = 2000000) {
  bool succeeded = false;
  bool failed = false;
  gTestController.SetCurrentTestCategory("weak");
  for (long count = 0; !succeeded && !failed && count < max_iters; ++count) {
    gTestController.BeginLeakDetection();
    {
      Value dup = v;
#ifndef EH_NO_EXCEPTIONS
      try {
#endif
        gTestController.SetFailureCountdown(count);
        op( dup );
        succeeded = true;
#ifndef EH_NO_EXCEPTIONS
      }
      catch (...) {}  // Just try again.
#endif
      gTestController.CancelFailureCountdown();
      CheckInvariant(dup);
    }
    failed = gTestController.ReportLeaked();
    EH_ASSERT( !failed );

    if ( succeeded )
      gTestController.ReportSuccess(count);
  }
  EH_ASSERT( succeeded || failed );  // Make sure the count hasn't gone over
}

/*===================================================================================
  ConstCheck

  EFFECTS:  Similar to WeakCheck (above), but for operations which may not modify
    their arguments. The operation is performed on the value itself, and no
    invariant checking is performed. Leak checking still occurs.
====================================================================================*/
template <class Value, class Operation>
void ConstCheck(const Value& v, const Operation& op, long max_iters = 2000000) {
  bool succeeded = false;
  bool failed = false;
  gTestController.SetCurrentTestCategory("const");
  for (long count = 0; !succeeded && !failed && count < max_iters; ++count) {
    gTestController.BeginLeakDetection();
    {
#ifndef EH_NO_EXCEPTIONS
      try {
#endif
        gTestController.SetFailureCountdown(count);
        op( v );
        succeeded = true;
#ifndef EH_NO_EXCEPTIONS
      }
      catch(...) {}  // Just try again.
# endif
      gTestController.CancelFailureCountdown();
    }
    failed = gTestController.ReportLeaked();
    EH_ASSERT( !failed );

    if ( succeeded )
      gTestController.ReportSuccess(count);
  }
  EH_ASSERT( succeeded || failed );  // Make sure the count hasn't gone over
}

/*===================================================================================
  StrongCheck

  EFFECTS:  Similar to WeakCheck (above), but additionally checks a component of
    the "strong guarantee": if the operation fails due to an exception, the
    value being operated on must be unchanged, as checked with operator==().

  CAVEATS: Note that this does not check everything required for the strong
    guarantee, which says that if an exception is thrown, the operation has no
    effects. Do do that we would have to check that no there were no side-effects
    on objects which are not part of v (e.g. iterator validity must be preserved).

====================================================================================*/
template <class Value, class Operation>
void StrongCheck(const Value& v, const Operation& op, long max_iters = 2000000) {
  bool succeeded = false;
  bool failed = false;
  gTestController.SetCurrentTestCategory("strong");
  for ( long count = 0; !succeeded && !failed && count < max_iters; count++ ) {
    gTestController.BeginLeakDetection();

    {
      Value dup = v;
      {
#ifndef EH_NO_EXCEPTIONS
        try {
#endif
          gTestController.SetFailureCountdown(count);
          op( dup );
          succeeded = true;
          gTestController.CancelFailureCountdown();
# ifndef EH_NO_EXCEPTIONS
        }
        catch (...) {
          gTestController.CancelFailureCountdown();
          bool unchanged = (dup == v);
          EH_ASSERT( unchanged );

          if ( !unchanged ) {
#if 0
            typedef typename Value::value_type value_type;
            EH_STD::ostream_iterator<value_type> o(EH_STD::cerr, " ");
            EH_STD::cerr<<"EH test FAILED:\nStrong guaranee failed !\n";
            EH_STD::copy(dup.begin(), dup.end(), o);
            EH_STD::cerr<<"\nOriginal is:\n";
            EH_STD::copy(v.begin(), v.end(), o);
            EH_STD::cerr<<EH_STD::endl;
#endif
            failed = true;
          }
        }  // Just try again.
# endif
        CheckInvariant(v);
      }
    }

    bool leaked = gTestController.ReportLeaked();
    EH_ASSERT( !leaked );
    if ( leaked )
      failed = true;

    if ( succeeded )
      gTestController.ReportSuccess(count);
  }
  EH_ASSERT( succeeded || failed );  // Make sure the count hasn't gone over
}

#endif // INCLUDED_MOTU_LeakCheck