// Copyright (C) 2003, International Business Machines
// Corporation and others.  All Rights Reserved.
#if defined(_MSC_VER)
// Turn off compiler warning about long names
#  pragma warning(disable:4786)
#endif

#include <cassert>

#include "CoinWarmStartDual.hpp"
#include <cmath>

//#############################################################################

/*
  Generate a `diff' that can convert the warm start passed as a parameter to
  the warm start specified by this.

  The capabilities are limited: the basis passed as a parameter can be no
  larger than the basis pointed to by this.
*/

CoinWarmStartDiff*
CoinWarmStartDual::generateDiff (const CoinWarmStart *const oldCWS) const
{ 
/*
  Make sure the parameter is CoinWarmStartDual or derived class.
*/
  const CoinWarmStartDual *oldDual =
      dynamic_cast<const CoinWarmStartDual *>(oldCWS) ;
  if (!oldDual)
  { throw CoinError("Old warm start not derived from CoinWarmStartDual.",
		    "generateDiff","CoinWarmStartDual") ; }
  const CoinWarmStartDual *newDual = this ;
/*
  Make sure newDual is equal or bigger than oldDual. Calculate the worst
  case number of diffs and allocate vectors to hold them.
*/
  const int oldCnt = oldDual->size() ;
  const int newCnt = newDual->size() ;

  assert(newCnt >= oldCnt) ;

  unsigned int *diffNdx = new unsigned int [newCnt]; 
  double *diffVal = new double [newCnt]; 
/*
  Scan the dual vectors.  For the portion of the vectors which overlap,
  create diffs. Then add any additional entries from newDual.
*/
  const double *oldVal = oldDual->dual() ;
  const double *newVal = newDual->dual() ;
  int numberChanged = 0 ;
  int i ;
  for (i = 0 ; i < oldCnt ; i++)
  { if (oldVal[i] != newVal[i])
    { diffNdx[numberChanged] = i ;
      diffVal[numberChanged++] = newVal[i] ; } }
  for ( ; i < newCnt ; i++)
  { diffNdx[numberChanged] = i ;
    diffVal[numberChanged++] = newVal[i] ; }
/*
  Create the object of our desire.
*/
  CoinWarmStartDualDiff *diff =
    new CoinWarmStartDualDiff(numberChanged,diffNdx,diffVal) ;
/*
  Clean up and return.
*/
  delete[] diffNdx ;
  delete[] diffVal ;

  return (dynamic_cast<CoinWarmStartDiff *>(diff)) ; }


/*
  Apply diff to this warm start.

  Update this warm start by applying diff. It's assumed that the
  allocated capacity of the warm start is sufficiently large.
*/

void CoinWarmStartDual::applyDiff (const CoinWarmStartDiff *const cwsdDiff)
{
/*
  Make sure we have a CoinWarmStartDualDiff
*/
  const CoinWarmStartDualDiff *diff =
    dynamic_cast<const CoinWarmStartDualDiff *>(cwsdDiff) ;
  if (!diff)
  { throw CoinError("Diff not derived from CoinWarmStartDualDiff.",
		    "applyDiff","CoinWarmStartDual") ; }
/*
  Application is by straighforward replacement of words in the dual vector.
*/
  const int numberChanges = diff->sze_ ;
  const unsigned int *diffNdxs = diff->diffNdxs_ ;
  const double *diffVals = diff->diffVals_ ;
  double *vals = this->dualVector_ ;

  for (int i = 0 ; i < numberChanges ; i++)
  { unsigned int diffNdx = diffNdxs[i] ;
    double diffVal = diffVals[i] ;
    vals[diffNdx] = diffVal ; }

  return ; }


//#############################################################################


// Assignment

CoinWarmStartDualDiff&
CoinWarmStartDualDiff::operator= (const CoinWarmStartDualDiff &rhs)

{ if (this != &rhs)
  { if (sze_ > 0)
    { delete[] diffNdxs_ ;
      delete[] diffVals_ ; }
    sze_ = rhs.sze_ ;
    if (sze_ > 0)
    { diffNdxs_ = new unsigned int[sze_] ;
      memcpy(diffNdxs_,rhs.diffNdxs_,sze_*sizeof(unsigned int)) ;
      diffVals_ = new double[sze_] ;
      memcpy(diffVals_,rhs.diffVals_,sze_*sizeof(double)) ; }
    else
    { diffNdxs_ = 0 ;
      diffVals_ = 0 ; } }

  return (*this) ; }


// Copy constructor

CoinWarmStartDualDiff::CoinWarmStartDualDiff (const CoinWarmStartDualDiff &rhs)
  : sze_(rhs.sze_),
    diffNdxs_(0),
    diffVals_(0)
{ if (sze_ > 0)
  { diffNdxs_ = new unsigned int[sze_] ;
    memcpy(diffNdxs_,rhs.diffNdxs_,sze_*sizeof(unsigned int)) ;
    diffVals_ = new double[sze_] ;
    memcpy(diffVals_,rhs.diffVals_,sze_*sizeof(double)) ; }
  
  return ; }


/// Standard constructor

CoinWarmStartDualDiff::CoinWarmStartDualDiff
  (int sze, const unsigned int *const diffNdxs, const double *const diffVals)
  : sze_(sze),
    diffNdxs_(0),
    diffVals_(0)
{ if (sze > 0)
  { diffNdxs_ = new unsigned int[sze] ;
    memcpy(diffNdxs_,diffNdxs,sze*sizeof(unsigned int)) ;
    diffVals_ = new double[sze] ;
    memcpy(diffVals_,diffVals,sze*sizeof(double)) ; }
  
  return ; }



syntax highlighted by Code2HTML, v. 0.9.1