// Class RooTrace controls the memory tracing hooks in all RooFit
// objects. When tracing is active, a table of live RooFit objects
// is kept that can be queried at any time. In verbose mode, messages
// are printed in addition at the construction and destruction of
// each object.
// END_HTML
#include "RooFit.h"
#include "RooTrace.h"
#include "RooAbsArg.h"
#include "Riostream.h"
#include "RooMsgService.h"
#include <iomanip>
using namespace std;
ClassImp(RooTrace)
;
RooTrace* RooTrace::_instance=0 ;
RooTrace& RooTrace::instance() 
{
  if (_instance==0) _instance = new RooTrace() ;
  return *_instance ;
}
RooTrace::RooTrace() : _active(kFALSE), _verbose(kFALSE) 
{
}
void RooTrace::create(const TObject* obj) 
{ 
  
  RooTrace& instance = RooTrace::instance() ;
  if (instance._active) {
    instance.create3(obj) ; 
  }
}
void RooTrace::destroy(const TObject* obj) 
{ 
  
  RooTrace& instance = RooTrace::instance() ;
  if (instance._active) {
    instance.destroy3(obj) ; 
  }
}
void RooTrace::createSpecial(const char* name, int size) 
{
  RooTrace& instance = RooTrace::instance() ;
  if (instance._active) {
    instance.createSpecial3(name,size) ;
  }
}
void RooTrace::destroySpecial(const char* name) 
{
  RooTrace& instance = RooTrace::instance() ;
  if (instance._active) {
    instance.destroySpecial3(name) ;
  }
}
void RooTrace::createSpecial3(const char* name, int size) 
{
  _specialCount[name]++ ;
  _specialSize[name] = size ;
}
void RooTrace::destroySpecial3(const char* name) 
{
  _specialCount[name]-- ;
}
void RooTrace::active(Bool_t flag) 
{ 
  
  
  RooTrace::instance()._active = flag ; 
}
void RooTrace::verbose(Bool_t flag) 
{ 
  
  
  RooTrace::instance()._verbose = flag ; 
}
void RooTrace::create2(const TObject* obj) 
{
  
  
  _list.Add((RooAbsArg*)obj) ;
  if (_verbose) {
    cout << "RooTrace::create: object " << obj << " of type " << obj->ClassName() 
	 << " created " << endl ;
  }
}
  
void RooTrace::destroy2(const TObject* obj) 
{
  
  if (!_list.Remove((RooAbsArg*)obj)) {
  } else if (_verbose) {
    cout << "RooTrace::destroy: object " << obj << " of type " << obj->ClassName() 
	 << " destroyed [" << obj->GetTitle() << "]" << endl ;
  }
}
void RooTrace::create3(const TObject* obj) 
{
  
  _objectCount[obj->IsA()]++ ;
}
  
void RooTrace::destroy3(const TObject* obj) 
{
  
  _objectCount[obj->IsA()]-- ;
}
void RooTrace::mark()
{
  
  
  RooTrace::instance().mark3() ;
}
void RooTrace::mark3()
{
  
  
  _markList = _list ;
}
void RooTrace::dump() 
{
  
  RooTrace::instance().dump3(cout,kFALSE) ;
}
void RooTrace::dump(ostream& os, Bool_t sinceMarked) 
{
  RooTrace::instance().dump3(os,sinceMarked) ;
}
void RooTrace::dump3(ostream& os, Bool_t sinceMarked) 
{
  
  
  os << "List of RooFit objects allocated while trace active:" << endl ;
  Int_t i, nMarked(0) ;
  for(i=0 ; i<_list.GetSize() ; i++) {
    if (!sinceMarked || _markList.IndexOf(_list.At(i)) == -1) {
      os << hex << setw(10) << _list.At(i) << dec << " : " << setw(20) << _list.At(i)->ClassName() << setw(0) << " - " << _list.At(i)->GetName() << endl ;
    } else {
      nMarked++ ;
    }
  }
  if (sinceMarked) os << nMarked << " marked objects suppressed" << endl ;
}
void RooTrace::printObjectCounts()
{
  RooTrace::instance().printObjectCounts3() ;
}
void RooTrace::printObjectCounts3() 
{
  Double_t total(0) ;
  for (map<TClass*,int>::iterator iter = _objectCount.begin() ; iter != _objectCount.end() ; ++iter) {
    Double_t tot= 1.0*(iter->first->Size()*iter->second)/(1024*1024) ;
    cout << " class " << iter->first->GetName() << " count = " << iter->second << " sizeof = " << iter->first->Size() << " total memory = " <<  Form("%5.2f",tot) << " Mb" << endl ;
    total+=tot ;
  }
  for (map<string,int>::iterator iter = _specialCount.begin() ; iter != _specialCount.end() ; ++iter) {
    int size = _specialSize[iter->first] ;
    Double_t tot=1.0*(size*iter->second)/(1024*1024) ;
    cout << " speeial " << iter->first << " count = " << iter->second << " sizeof = " << size  << " total memory = " <<  Form("%5.2f",tot) << " Mb" << endl ;
    total+=tot ;
  }
  cout << "Grand total memory = " << Form("%5.2f",total) << " Mb" << endl ;
}
void RooTrace::callgrind_zero() 
{
  
  
  
  
  
  
  ooccoutD((TObject*)0,Tracing) << "RooTrace::callgrind_zero()" << endl ;
}
void RooTrace::callgrind_dump() 
{
  
  
  
  
  
  ooccoutD((TObject*)0,Tracing) << "RooTrace::callgrind_dump()" << endl ;
}