#include "THaVarList.h"
#include "THaVar.h"
#include "THaRTTI.h"
#include "TRegexp.h"
#include "TString.h"
#include "TClass.h"
#include "TMethodCall.h"
#include "TROOT.h"
#include "TMath.h"
#include <algorithm>
#include <cstring>
ClassImp(THaVarList)
void THaVarList::Clear( Option_t* )
{
while( fFirst ) {
THaVar* obj = (THaVar*)TList::Remove( fFirst );
delete obj;
}
}
THaVar* THaVarList::DefineByType( const char* name, const char* descript,
const void* var, VarType type,
const Int_t* count )
{
static const char* const here = "DefineByType()";
THaVar* ptr = Find( name );
if( ptr ) {
Warning( here, "Variable %s already exists. Not redefined.",
ptr->GetName() );
return NULL;
}
ptr = new THaVar( name, descript, var, type, -1, NULL, count );
if( ptr )
AddLast( ptr );
else
Error( here, "Error allocating new variable %s. No variable defined.",
name );
return ptr;
}
THaVar* THaVarList::DefineByRTTI( const TString& name, const TString& desc,
const TString& def, const void* const obj,
TClass* const cl, const char* errloc )
{
if( !obj || !cl ) {
Warning( errloc, "Invalid class or object. Variable %s not defined.",
name.Data() );
return NULL;
}
THaVar* var = Find( name );
if( var ) {
Warning( errloc, "Variable %s already exists. Not redefined.",
var->GetName() );
return NULL;
}
Ssiz_t dot[2], pos = 0, ppos = 0;
Int_t ndot = 0;
TString s[3];
while( (pos = def.Index( ".", pos )) != kNPOS && ndot<2 ) {
dot[ndot] = pos;
Int_t len = pos - ppos;
if( len >= 0 )
s[ndot] = def(ppos, len);
ppos = ++pos;
++ndot;
}
s[ndot] = def(ppos, def.Length()-ppos );
if( pos != kNPOS ) {
Warning( errloc, "Too many dots in definition of variable %s (%s). "
"Variable not defined.", name.Data(), desc.Data() );
return NULL;
}
Int_t i = 0;
while( i <= ndot )
if( s[i++].Length() == 0 )
return NULL;
Bool_t function = s[ndot].EndsWith("()");
TMethodCall* theMethod = NULL;
TClass* theClass = NULL;
ULong_t loc = (ULong_t)obj;
void** ploc;
THaRTTI rtti, objrtti;
if( !function ) {
switch( ndot ) {
case 0:
rtti.Find( cl, s[0], obj );
break;
case 1:
objrtti.Find( cl, s[0], obj );
if( !objrtti.IsValid() || !objrtti.IsObject() )
return NULL;
theClass = objrtti.GetClass();
if( !theClass ) return NULL;
loc += objrtti.GetOffset();
if( objrtti.IsPointer() ) {
ploc = (void**)loc;
loc = (ULong_t)*ploc;
}
rtti.Find( theClass, s[1] );
break;
case 2:
objrtti.Find( cl, s[0], obj );
if( !objrtti.IsValid() || !objrtti.IsObject() )
return NULL;
theClass = objrtti.GetClass();
if( !theClass ) return NULL;
if( !theClass->InheritsFrom( "TSeqCollection" )) {
Warning( errloc, "Data member %s is not a TSeqCollection. "
"Variable %s (%s) not defined.",
s[0].Data(), name.Data(), desc.Data() );
return NULL;
}
loc += objrtti.GetOffset();
if( objrtti.IsPointer() ) {
ploc = (void**)loc;
loc = (ULong_t)*ploc;
}
theClass = gROOT->GetClass( s[1] );
if( !theClass ) {
Warning( errloc, "Cannot determine class of container "
"member object %s. Variable %s (%s) not defined.",
s[1].Data(), name.Data(), desc.Data() );
return NULL;
}
rtti.Find( theClass, s[2] );
break;
default:
break;
}
if( !rtti.IsValid() ) {
Warning( errloc, "No RTTI information for variable %s. "
"Not defined.", name.Data() );
return NULL;
}
if( rtti.IsObject() ) {
Warning( errloc, "Variable %s is an object. Must be a basic type. "
"Not defined.", name.Data() );
return NULL;
}
if( ndot < 2 ) {
loc += rtti.GetOffset();
TString theName(name);
Int_t* count_loc = NULL;
switch( rtti.GetArrayType() ) {
case THaRTTI::kFixed:
theName.Append( rtti.GetSubscript() );
break;
case THaRTTI::kVariable:
count_loc = (Int_t*)( (ULong_t)obj + rtti.GetCountOffset() );
break;
default:
break;
}
var = DefineByType( theName, desc, (void*)loc, rtti.GetType(), count_loc );
} else {
var = new THaVar( name, desc, (void*)loc, rtti.GetType(),
rtti.GetOffset() );
if( var )
AddLast( var );
else {
Error( errloc, "Error allocating new variable %s. No variable defined.",
name.Data() );
return NULL;
}
}
} else {
TString funcName(s[ndot]);
Ssiz_t i = funcName.Index( '(' );
if( i == kNPOS )
return NULL;
funcName = funcName(0,i);
switch( ndot ) {
case 0:
theMethod = new TMethodCall( cl, funcName, "" );
break;
case 1:
objrtti.Find( cl, s[0], obj );
if( !objrtti.IsValid() || !objrtti.IsObject() )
return NULL;
theClass = objrtti.GetClass();
if( !theClass ) return NULL;
loc += objrtti.GetOffset();
if( objrtti.IsPointer() ) {
ploc = (void**)loc;
loc = (ULong_t)*ploc;
}
theMethod = new TMethodCall( theClass, funcName, "" );
break;
case 2:
objrtti.Find( cl, s[0], obj );
if( !objrtti.IsValid() || !objrtti.IsObject() )
return NULL;
theClass = objrtti.GetClass();
if( !theClass ) return NULL;
if( !theClass->InheritsFrom( "TSeqCollection" )) {
Warning( errloc, "Data member %s is not a TSeqCollection. "
"Variable %s (%s) not defined.",
s[0].Data(), name.Data(), desc.Data() );
return NULL;
}
loc += objrtti.GetOffset();
if( objrtti.IsPointer() ) {
ploc = (void**)loc;
loc = (ULong_t)*ploc;
}
theClass = gROOT->GetClass( s[1] );
if( !theClass ) {
Warning( errloc, "Cannot determine class of container "
"member object %s. Variable %s (%s) not defined.",
s[1].Data(), name.Data(), desc.Data() );
return NULL;
}
theMethod = new TMethodCall( theClass, funcName, "" );
break;
default:
break;
}
if( !theMethod ) {
Warning( errloc, "Error getting function information for variable %s. "
"Not defined.", name.Data() );
return NULL;
}
if( !theMethod->GetMethod() ) {
Warning( errloc, "Function %s does not exist. "
"Variable %s not defined.", s[ndot].Data(), name.Data() );
delete theMethod;
return NULL;
}
VarType type;
switch( theMethod->ReturnType()) {
case TMethodCall::kLong:
type = kLong;
break;
case TMethodCall::kDouble:
type = kDouble;
break;
default:
Warning( errloc, "Undefined return type for function %s. "
"Variable %s not defined.", s[ndot].Data(), name.Data() );
return NULL;
delete theMethod;
break;
}
var = new THaVar( name, desc, (void*)loc, type, ((ndot==2) ? 0 : -1),
theMethod);
if( var )
AddLast( var );
else {
Error( errloc, "Error allocating new variable %s. No variable defined.",
name.Data() );
delete theMethod;
return NULL;
}
}
return var;
}
Int_t THaVarList::DefineVariables( const VarDef* list, const char* prefix,
const char* caller )
{
TString errloc("DefineVariables()");
if( caller) {
errloc.Append(" ");
errloc.Append(caller);
}
if( !list ) {
Warning(errloc, "Empty input list. No variables registered.");
return -1;
}
const VarDef* item = list;
Int_t ndef = 0;
size_t nlen = 64;
char* name = new char[nlen];
while( item->name ) {
const char* description = item->desc;
if( !description || !*description )
description = item->name;
size_t len = strlen( item->name ), slen = 0, plen = 0;
if( item->size>1 ) {
slen = static_cast<size_t>( TMath::Log10( item->size )) + 3;
}
if( prefix ) {
plen = strlen( prefix );
}
if( plen+len+slen+1 > nlen ) {
delete [] name;
name = new char[plen+len+slen+1];
nlen = plen+len+slen+1;
}
if( plen )
strcpy( name, prefix );
strcpy( name+plen, item->name );
if( slen ) {
sprintf( name+plen+len, "[%d]", item->size );
}
THaVar* var = DefineByType( name, description, item->loc,
static_cast<VarType>( item->type ),
item->count );
if( var )
ndef++;
item++;
}
delete [] name;
return ndef;
}
Int_t THaVarList::DefineVariables( const RVarDef* list, const TObject* obj,
const char* prefix, const char* caller,
const char* var_prefix )
{
TString errloc("DefineVariables()");
if( caller) {
errloc.Append(" ");
errloc.Append(caller);
}
if( !list ) {
Warning(errloc, "No input list. No variables registered.");
return -1;
}
if( !obj ) {
Warning(errloc, "No base object. No variables registered.");
return -2;
}
TClass* cl = obj->IsA();
if( !cl ) {
Warning( errloc, "Base object has no class?!? No variables registered.");
return -3;
}
const RVarDef* item;
Int_t ndef = 0;
while( (item = list++) && item->name ) {
TString name( item->name );
if( name.IsNull() )
continue;
if( prefix )
name.Prepend(prefix);
TString desc( (item->desc && *item->desc ) ? item->desc : name.Data() );
char* c = ::Compress( item->def );
TString def( c );
delete [] c;
if( def.IsNull() ) {
Warning( errloc, "Invalid definition for variable %s (%s). "
"Variable not defined.", name.Data(), desc.Data() );
continue;
}
if( var_prefix && *var_prefix )
def.Prepend( var_prefix );
THaVar* var = DefineByRTTI( name, desc, def, obj, cl, errloc );
if( var )
ndef++;
}
return ndef;
}
THaVar* THaVarList::Find( const char* name ) const
{
THaVar* ptr;
const char* p = strchr( name, '[' );
if( !p )
return static_cast<THaVar*>(FindObject( name ));
else {
size_t n = p-name;
char* basename = new char[ n+1 ];
strncpy( basename, name, n );
basename[n] = '\0';
ptr = static_cast<THaVar*>(FindObject( basename ));
delete [] basename;
}
return ptr;
}
void THaVarList::PrintFull( Option_t* option ) const
{
if( !option ) option = "";
TRegexp re(option,kTRUE);
TIter next(this);
TObject* object;
while ((object = next())) {
TString s = object->GetName();
if (*option && strcmp(option,object->GetName()) && s.Index(re) == kNPOS)
continue;
object->Print("FULL");
}
}
Int_t THaVarList::RemoveName( const char* name )
{
THaVar* ptr = Find( name );
if( ptr ) {
THaVar* p = static_cast< THaVar* >(TList::Remove( ptr ));
delete p;
return 1;
} else
return 0;
}
Int_t THaVarList::RemoveRegexp( const char* expr, Bool_t wildcard )
{
TRegexp re( expr, wildcard );
if( re.Status() ) return -1;
Int_t ndel = 0;
TString name;
TIter next( this );
while( THaVar* ptr = static_cast< THaVar* >( next() )) {
name = ptr->GetName();
if( name.Index( re ) != kNPOS ) {
THaVar* p = static_cast< THaVar* >(TList::Remove( ptr ));
delete p;
ndel++;
}
}
return ndel;
}
Last change: Sat Nov 7 21:26:56 2009
Last generated: 2009-11-07 21:26
This page has been automatically generated. If you have any comments or suggestions about the page layout send a mail to ROOT support, or contact the developers with any questions or problems regarding ROOT.