#include "THaAnalysisObject.h"
#include "THaVarList.h"
#include "TClass.h"
#include "TDatime.h"
#include "TROOT.h"
#include "TMath.h"
#include "TError.h"
#include "TVector3.h"
#include "TSystem.h"
#include <cstring>
#include <cctype>
#include <errno.h>
#include <algorithm>
#include <iostream>
#include <cmath>
#include <cstdlib>
#include <cstdio>
#include <string>
#ifdef HAS_SSTREAM
#include <sstream>
#define ISSTREAM istringstream
#else
#include <strstream>
#define ISSTREAM istrstream
#endif
using namespace std;
typedef string::size_type ssiz_t;
TList* THaAnalysisObject::fgModules = NULL;
const Double_t THaAnalysisObject::kBig = 1.e38;
THaAnalysisObject::THaAnalysisObject( const char* name,
const char* description ) :
TNamed(name,description), fPrefix(NULL), fStatus(kNotinit),
fDebug(0), fIsInit(false), fIsSetup(false), fProperties(0),
fOKOut(false), fInitDate(19950101,0)
{
if( !fgModules ) fgModules = new TList;
fgModules->Add( this );
}
THaAnalysisObject::THaAnalysisObject( ) : fPrefix(NULL), fIsSetup(false) {
}
THaAnalysisObject::~THaAnalysisObject()
{
if (fgModules) {
fgModules->Remove( this );
if( fgModules->GetSize() == 0 ) {
delete fgModules;
fgModules = 0;
}
}
delete [] fPrefix; fPrefix = 0;
}
Int_t THaAnalysisObject::Begin( THaRunBase* )
{
return 0;
}
Int_t THaAnalysisObject::DefineVariables( EMode )
{
return kOK;
}
Int_t THaAnalysisObject::DefineVarsFromList( const VarDef* list, EMode mode,
const char* var_prefix ) const
{
return DefineVarsFromList( list, kVarDef, mode, var_prefix );
}
Int_t THaAnalysisObject::DefineVarsFromList( const RVarDef* list, EMode mode,
const char* var_prefix ) const
{
return DefineVarsFromList( list, kRVarDef, mode, var_prefix );
}
Int_t THaAnalysisObject::DefineVarsFromList( const void* list,
EType type, EMode mode,
const char* var_prefix ) const
{
return DefineVarsFromList( list, type, mode, var_prefix, this,
fPrefix, Here(ClassName()) );
}
Int_t THaAnalysisObject::DefineVarsFromList( const void* list,
EType type, EMode mode,
const char* var_prefix,
const TObject* obj,
const char* prefix,
const char* here )
{
if( !gHaVars ) {
TString action;
if( mode == kDefine )
action = "defined";
else if( mode == kDelete )
action = "deleted (this is safe when exiting)";
::Warning( here, "No global variable list found. No variables %s.",
action.Data() );
return (mode==kDefine ? kInitError : kOK);
}
if( mode == kDefine ) {
if( type == kVarDef )
gHaVars->DefineVariables( (const VarDef*)list, prefix, here );
else if( type == kRVarDef )
gHaVars->DefineVariables( (const RVarDef*)list, obj,
prefix, here, var_prefix );
}
else if( mode == kDelete ) {
if( type == kVarDef ) {
const VarDef* item;
const VarDef* theList = (const VarDef*)list;
while( (item = theList++) && item->name ) {
TString name(prefix);
name.Append( item->name );
gHaVars->RemoveName( name );
}
} else if( type == kRVarDef ) {
const RVarDef* item;
const RVarDef* theList = (const RVarDef*)list;
while( (item = theList++) && item->name ) {
TString name(prefix);
name.Append( item->name );
gHaVars->RemoveName( name );
}
}
}
return kOK;
}
Int_t THaAnalysisObject::End( THaRunBase* )
{
return 0;
}
THaAnalysisObject* THaAnalysisObject::FindModule( const char* name,
const char* classname,
bool do_error )
{
static const char* const here = "FindModule()";
static const char* const anaobj = "THaAnalysisObject";
EStatus save_status = fStatus;
if( do_error )
fStatus = kInitError;
if( !name || !*name ) {
if( do_error )
Error( Here(here), "No module name given." );
return NULL;
}
TObject* obj = fgModules->FindObject( name );
if( !obj ) {
if( do_error )
Error( Here(here), "Module %s does not exist.", name );
return NULL;
}
if( !obj->IsA()->InheritsFrom( anaobj )) {
if( do_error )
Error( Here(here), "Module %s (%s) is not a %s.",
obj->GetName(), obj->GetTitle(), anaobj );
return NULL;
}
if( classname && *classname && strcmp(classname,anaobj) &&
!obj->IsA()->InheritsFrom( classname )) {
if( do_error )
Error( Here(here), "Module %s (%s) is not a %s.",
obj->GetName(), obj->GetTitle(), classname );
return NULL;
}
THaAnalysisObject* aobj = static_cast<THaAnalysisObject*>( obj );
if( do_error ) {
if( !aobj->IsOK() ) {
Error( Here(here), "Module %s (%s) not initialized.",
obj->GetName(), obj->GetTitle() );
return NULL;
}
}
fStatus = save_status;
return aobj;
}
vector<string> THaAnalysisObject::GetDBFileList( const char* name,
const TDatime& date,
const char* here )
{
const char* dbdir = NULL;
const char* result;
void* dirp;
size_t pos;
vector<string> time_dirs, dnames, fnames;
vector<string>::iterator it, thedir;
string item, filename;
Int_t item_date;
const string defaultdir("DEFAULT");
bool have_defaultdir = false, found = false;
if( (dbdir = gSystem->Getenv("DB_DIR")))
dnames.push_back( dbdir );
dnames.push_back( "DB" );
dnames.push_back( "db" );
dnames.push_back( "." );
it = dnames.begin();
while( !(dirp = gSystem->OpenDirectory( (*it).c_str() )) &&
(++it != dnames.end()) ) {}
if( it == dnames.end() ) {
::Error( here, "Cannot open any database directories. Check your disk!");
goto exit;
}
thedir = it;
while( (result = gSystem->GetDirEntry(dirp)) ) {
item = result;
if( item.length() == 8 ) {
for( pos=0; pos<8; ++pos )
if( !isdigit(item[pos])) break;
if( pos==8 )
time_dirs.push_back( item );
} else if ( item == defaultdir )
have_defaultdir = true;
}
gSystem->FreeDirectory(dirp);
if( time_dirs.size() > 0 ) {
sort( time_dirs.begin(), time_dirs.end() );
for( it = time_dirs.begin(); it != time_dirs.end(); it++ ) {
item_date = atoi((*it).c_str());
if( it == time_dirs.begin() && date.GetDate() < item_date )
break;
if( it != time_dirs.begin() && date.GetDate() < item_date ) {
it--;
found = true;
break;
}
if( it+1 == time_dirs.end() && date.GetDate() >= item_date ) {
found = true;
break;
}
}
}
filename = "db_";
filename += name;
if( *filename.rbegin() != '.' )
filename += '.';
filename += "dat";
fnames.push_back( filename );
if( found ) {
item = *thedir + "/" + *it + "/" + filename;
fnames.push_back( item );
}
if( have_defaultdir ) {
item = *thedir + "/" + defaultdir + "/" + filename;
fnames.push_back( item );
}
fnames.push_back( *thedir + "/" + filename );
exit:
return fnames;
}
const char* THaAnalysisObject::GetDBFileName() const
{
return GetPrefix();
}
const char* Here( const char* here, const char* prefix )
{
static char buf[256];
buf[0] = 0;
if(prefix != NULL) {
strcpy( buf, "\"" );
strcat( buf, prefix );
*(buf+strlen(buf)-1) = 0;
strcat( buf, "\"::" );
}
strcat( buf, here );
return buf;
}
const char* THaAnalysisObject::Here( const char* here ) const
{
return ::Here( here, fPrefix );
}
THaAnalysisObject::EStatus THaAnalysisObject::Init()
{
return Init( TDatime() );
}
THaAnalysisObject::EStatus THaAnalysisObject::Init( const TDatime& date )
{
static const char* const here = "Init";
if( IsZombie() )
return fStatus = kNotinit;
fInitDate = date;
Int_t status = kOK;
const char* fnam = "run.";
MakePrefix();
status = ReadRunDatabase(date);
if( status && (status != kFileError || (fProperties & kNeedsRunDB) != 0))
goto err;
if( IsA()->GetMethodAllAny("ReadDatabase")
!= gROOT->GetClass("THaAnalysisObject")->GetMethodAllAny("ReadDatabase")){
fnam = GetDBFileName();
if( (status = ReadDatabase(date)) )
goto err;
}
else if ( fDebug>0 ) {
cout << "Info: Skipping database call for object " << GetName()
<< " since no ReadDatabase function defined.\n";
}
status = DefineVariables(kDefine);
Clear("I");
goto exit;
err:
if( status == kFileError )
Error(Here(here),"Cannot open database file db_%sdat",fnam);
else if( status == kInitError )
Error(Here(here),"Error when reading file db_%sdat",fnam);
exit:
return fStatus = (EStatus)status;
}
Int_t THaAnalysisObject::InitOutput( THaOutput* )
{
fOKOut = true;
return kOK;
}
void THaAnalysisObject::MakePrefix( const char* basename )
{
delete [] fPrefix;
if( basename && *basename ) {
fPrefix = new char[ strlen(basename) + strlen(GetName()) + 3 ];
strcpy( fPrefix, basename );
strcat( fPrefix, "." );
} else {
fPrefix = new char[ strlen(GetName()) + 2 ];
*fPrefix = 0;
}
strcat( fPrefix, GetName() );
strcat( fPrefix, "." );
}
FILE* THaAnalysisObject::OpenFile( const char *name, const TDatime& date,
const char *here, const char *filemode,
const int debug_flag )
{
if( !name || !*name )
return NULL;
if( !here )
here="";
if( !filemode )
filemode="r";
FILE* fi = NULL;
vector<string> fnames( GetDBFileList(name, date, here) );
if( !fnames.empty() ) {
vector<string>::iterator it = fnames.begin();
do {
if( debug_flag>1 )
cout << "<" << here << ">: Opening database file " << *it;
fi = fopen( (*it).c_str(), filemode);
if( debug_flag>1 )
if( !fi ) cout << " ... failed" << endl;
else cout << " ... ok" << endl;
else if( debug_flag>0 && fi )
cout << "<" << here << ">: Opened database file " << *it << endl;
} while ( !fi && ++it != fnames.end() );
}
if( !fi && debug_flag>0 ) {
::Error(here,"Cannot open database file db_%s%sdat",name,
(name[strlen(name)-1]=='.'?"":"."));
}
return fi;
}
FILE* THaAnalysisObject::OpenFile( const TDatime& date )
{
return OpenFile(GetDBFileName(), date, Here("OpenFile()"), "r", fDebug);
}
FILE* THaAnalysisObject::OpenRunDBFile( const TDatime& date )
{
return OpenFile("run", date, Here("OpenFile()"), "r", fDebug);
}
char* THaAnalysisObject::ReadComment( FILE* fp, char *buf, const int len )
{
int ch = fgetc(fp);
ungetc(ch,fp);
if (ch == EOF || ch == ' ')
return NULL;
char *s= fgets(buf,len,fp);
return s;
}
Int_t THaAnalysisObject::ReadDatabase( const TDatime& )
{
return kOK;
}
Int_t THaAnalysisObject::ReadRunDatabase( const TDatime& date )
{
if( !fPrefix ) return kInitError;
FILE* file = OpenRunDBFile( date );
if( !file ) return kFileError;
TString name(fPrefix); name.Append("config");
Int_t ret = LoadDBvalue( file, date, name, fConfig );
fclose(file);
if( ret == -1 ) return kFileError;
if( ret < 0 ) return kInitError;
return kOK;
}
Int_t THaAnalysisObject::RemoveVariables()
{
return DefineVariables( kDelete );
}
void THaAnalysisObject::SetName( const char* name )
{
if( !name || !*name ) {
Warning( Here("SetName()"),
"Cannot set an empty object name. Name not set.");
return;
}
TNamed::SetName( name );
MakePrefix();
}
void THaAnalysisObject::SetNameTitle( const char* name, const char* title )
{
SetName( name );
SetTitle( title );
}
void THaAnalysisObject::SetConfig( const char* label )
{
fConfig = label;
}
void THaAnalysisObject::SetDebug( Int_t level )
{
fDebug = level;
}
Bool_t THaAnalysisObject::IntersectPlaneWithRay( const TVector3& xax,
const TVector3& yax,
const TVector3& org,
const TVector3& ray_start,
const TVector3& ray_vect,
Double_t& length,
TVector3& intersect )
{
Double_t nom[9], den[9];
nom[0] = den[0] = xax.X();
nom[3] = den[3] = xax.Y();
nom[6] = den[6] = xax.Z();
nom[1] = den[1] = yax.X();
nom[4] = den[4] = yax.Y();
nom[7] = den[7] = yax.Z();
den[2] = -ray_vect.X();
den[5] = -ray_vect.Y();
den[8] = -ray_vect.Z();
Double_t det1 = den[0]*(den[4]*den[8]-den[7]*den[5])
-den[3]*(den[1]*den[8]-den[7]*den[2])
+den[6]*(den[1]*den[5]-den[4]*den[2]);
if( fabs(det1) < 1e-5 )
return false;
nom[2] = ray_start.X()-org.X();
nom[5] = ray_start.Y()-org.Y();
nom[8] = ray_start.Z()-org.Z();
Double_t det2 = nom[0]*(nom[4]*nom[8]-nom[7]*nom[5])
-nom[3]*(nom[1]*nom[8]-nom[7]*nom[2])
+nom[6]*(nom[1]*nom[5]-nom[4]*nom[2]);
length = det2/det1;
intersect = ray_start + length*ray_vect;
return true;
}
void THaAnalysisObject::GeoToSph( Double_t th_geo, Double_t ph_geo,
Double_t& th_sph, Double_t& ph_sph )
{
static const Double_t twopi = 2.0*TMath::Pi();
Double_t ct = cos(th_geo), cp = cos(ph_geo);
register Double_t tmp = ct*cp;
th_sph = acos( tmp );
tmp = sqrt(1.0 - tmp*tmp);
ph_sph = (fabs(tmp) < 1e-6 ) ? 0.0 : acos( sqrt(1.0-ct*ct)*cp/tmp );
if( th_geo/twopi-floor(th_geo/twopi) > 0.5 ) ph_sph = TMath::Pi() - ph_sph;
if( ph_geo/twopi-floor(ph_geo/twopi) > 0.5 ) ph_sph = -ph_sph;
}
void THaAnalysisObject::SphToGeo( Double_t th_sph, Double_t ph_sph,
Double_t& th_geo, Double_t& ph_geo )
{
static const Double_t twopi = 2.0*TMath::Pi();
Double_t ct = cos(th_sph), st = sin(th_sph), cp = cos(ph_sph);
if( fabs(ct) > 1e-6 ) {
th_geo = atan( st/ct*cp );
if( cp>0.0 && th_geo<0.0 ) th_geo += TMath::Pi();
else if( cp<0.0 && th_geo>0.0 ) th_geo -= TMath::Pi();
} else {
th_geo = TMath::Pi()/2.0;
if( cp<0.0 ) th_geo = -th_geo;
}
ph_geo = acos( sqrt( st*st*cp*cp + ct*ct ));
if( ph_sph/twopi - floor(ph_sph/twopi) > 0.5 ) ph_geo =- ph_geo;
}
static size_t LEN = 256;
static size_t MAX = 1UL<<14;
static string errtxt;
static int loaddb_depth = 0;
static string loaddb_prefix;
static Int_t IsDBdate( const string& line, TDatime& date, bool warn = true )
{
ssiz_t lbrk = line.find('[');
if( lbrk == string::npos || lbrk >= line.size()-12 ) return 0;
ssiz_t rbrk = line.find(']',lbrk);
if( rbrk == string::npos || rbrk <= lbrk+11 ) return 0;
Int_t yy, mm, dd, hh, mi, ss;
if( sscanf( line.substr(lbrk+1,rbrk-lbrk-1).c_str(), "%d-%d-%d %d:%d:%d",
&yy, &mm, &dd, &hh, &mi, &ss) != 6
|| yy < 1995 || mm < 1 || mm > 12 || dd < 1 || dd > 31
|| hh < 0 || hh > 23 || mi < 0 || mi > 59 || ss < 0 || ss > 59 ) {
if( warn )
::Warning("THaAnalysisObject::IsDBdate()",
"Invalid date tag %s", line.c_str());
return 0;
}
date.Set(yy, mm, dd, hh, mi, ss);
return 1;
}
static Int_t TrimDBline( const string& _line, string& value,
bool leading = false )
{
value.erase();
string line(_line);
ssiz_t pos1 = line.find("#");
if( pos1 != string::npos )
line.erase(pos1);
pos1 = (leading) ? line.find_first_not_of(" \t") : 0;
if( pos1 == string::npos ) return 1;
ssiz_t pos2 = line.find_last_not_of(" \t");
value = line.substr(pos1,pos2-pos1+1);
if( line[pos2] == '\\' ) {
value.erase(pos2-pos1);
return 2;
}
return 1;
}
static Int_t IsDBkey( const string& line, const char* key, string& text )
{
ssiz_t pos = line.find('=');
if( pos == string::npos ) return 0;
if( pos == 0 || pos == line.size()-1 ) return -1;
ssiz_t pos1 = line.substr(0,pos).find_first_not_of(" \t");
if( pos1 == string::npos ) return -1;
ssiz_t pos2 = line.substr(0,pos).find_last_not_of(" \t");
if( pos2 == string::npos ) return -1;
string lhs(line.substr(pos1,pos2-pos1+1).c_str());
if( lhs != key ) return -1;
string rhs = line.substr(pos+1);
if( rhs.find_first_not_of(" \t") == string::npos ) return -2;
return TrimDBline( rhs, text, true );
}
static bool IsTag( const char* buf )
{
register const char* p = buf;
while( isspace(*p)) p++;
if( *p != '[' ) return false;
while( *p && *p != ']' ) p++;
return ( *p == ']' );
}
static Int_t ChopPrefix( string& s )
{
ssiz_t len = s.size(), pos;
Int_t ndot = 0;
if( len<2 )
goto null;
pos = s.rfind('.',len-2);
if( pos == string::npos )
goto null;
s.erase(pos+1);
for( ssiz_t i = 0; i <= pos; i++ ) {
if( s[i] == '.' )
ndot++;
}
return ndot;
null:
s.clear();
return 0;
}
Int_t THaAnalysisObject::LoadDBvalue( FILE* file, const TDatime& date,
const char* key, string& text )
{
if( !file || !key ) return -255;
char* buf = new char[LEN];
if( !buf ) return -256;
TDatime keydate(950101,0), prevdate(950101,0);
errno = 0;
errtxt.erase();
rewind(file);
bool found = false, ignore = false, multi_line = false, not_prev_cont = true;
off_t pos = ftello(file);
while( fgets( buf, LEN, file) != NULL) {
size_t len = strlen(buf);
if( len >= LEN-1 ) {
LEN *= 2;
if( LEN > MAX ) {
buf[72] = 0;
errtxt = buf;
}
delete [] buf;
if( LEN > MAX )
return -128;
buf = new char[LEN];
if( !buf ) return -256;
fseeko( file, pos, SEEK_SET );
continue;
}
pos = ftello(file);
if( len == 0 ) continue;
if( buf[len-1] == '\n') buf[len-1] = 0;
string line(buf), value;
Int_t status = TrimDBline( line, value ), cur_status = status;
if( len < 2 || ( !multi_line && status == 1 && value.empty() )) {
not_prev_cont = true;
multi_line = false;
continue;
}
if( multi_line && status == 1 && value.empty() &&
line.find('#') != string::npos )
continue;
if( !ignore && ( multi_line || ( not_prev_cont &&
(status = IsDBkey( line, key, value )) != 0 ) )) {
if( status > 0 ) {
if( multi_line ) {
text.append(value);
} else {
found = true;
prevdate = keydate;
text = value;
}
multi_line = ( status == 2 );
}
} else if( not_prev_cont && IsDBdate( line, keydate ) != 0 )
ignore = ( keydate>date || keydate<prevdate );
not_prev_cont = ( cur_status != 2 );
}
delete [] buf;
if( errno ) {
perror( "THaAnalysisObject::LoadDBvalue" );
return -1;
}
return found ? 0 : 1;
}
Int_t THaAnalysisObject::LoadDBvalue( FILE* file, const TDatime& date,
const char* key, Double_t& value )
{
string text;
Int_t err = LoadDBvalue( file, date, key, text );
if( err == 0 )
value = atof(text.c_str());
return err;
}
Int_t THaAnalysisObject::LoadDBvalue( FILE* file, const TDatime& date,
const char* key, Int_t& value )
{
string text;
Int_t err = LoadDBvalue( file, date, key, text );
if( err == 0 )
value = atoi(text.c_str());
return err;
}
Int_t THaAnalysisObject::LoadDBvalue( FILE* file, const TDatime& date,
const char* key, TString& text )
{
string _text;
Int_t err = LoadDBvalue( file, date, key, _text );
if( err == 0 )
text = _text.c_str();
return err;
}
template <class T>
Int_t THaAnalysisObject::LoadDBarray( FILE* file, const TDatime& date,
const char* key, vector<T>& values )
{
string text;
Int_t err = LoadDBvalue( file, date, key, text );
if( err )
return err;
values.clear();
text += " ";
ISSTREAM inp(text);
T dval;
while( 1 ) {
inp >> dval;
if( inp.good() )
values.push_back(dval);
else
break;
}
return 0;
}
template <class T>
Int_t THaAnalysisObject::LoadDBmatrix( FILE* file, const TDatime& date,
const char* key,
vector<vector<T> >& values,
UInt_t ncols )
{
vector<T>* tmpval = new vector<T>;
if( !tmpval )
return -255;
Int_t err = LoadDBarray( file, date, key, *tmpval );
if( err ) {
delete tmpval;
return err;
}
if( (tmpval->size() % ncols) != 0 ) {
delete tmpval;
errtxt = "key = "; errtxt += key;
return -129;
}
values.clear();
typename vector<vector<T> >::size_type nrows = tmpval->size()/ncols, irow;
for( irow = 0; irow < nrows; ++irow ) {
vector<T> row;
for( typename vector<T>::size_type i=0; i<ncols; ++i ) {
row.push_back( tmpval->at(i+irow*ncols) );
}
values.push_back( row );
}
delete tmpval;
return 0;
}
Int_t THaAnalysisObject::LoadDB( FILE* f, const TDatime& date,
const DBRequest* req, const char* prefix,
Int_t search )
{
const char* const here = "THaAnalysisObject::LoadDB";
if( !req ) return -255;
if( !prefix ) prefix = "";
Int_t ret = 0;
if( loaddb_depth++ == 0 )
loaddb_prefix = prefix;
const DBRequest* item = req;
while( item->name ) {
if( item->var ) {
string keystr(prefix); keystr.append(item->name);
const char* key = keystr.c_str();
if( item->type == kDouble || item->type == kFloat ) {
Double_t dval = 0.0;
ret = LoadDBvalue( f, date, key, dval );
if( ret == 0 ) {
if( item->type == kDouble )
*((Double_t*)item->var) = dval;
else
*((Float_t*)item->var) = dval;
}
} else if( item->type >= kInt && item->type <= kByte ) {
Int_t ival;
ret = LoadDBvalue( f, date, key, ival );
if( ret == 0 ) {
switch( item->type ) {
case kInt:
*((Int_t*)item->var) = ival;
break;
case kUInt:
*((UInt_t*)item->var) = ival;
break;
case kShort:
*((Short_t*)item->var) = ival;
break;
case kUShort:
*((UShort_t*)item->var) = ival;
break;
case kChar:
*((Char_t*)item->var) = ival;
break;
case kByte:
*((Byte_t*)item->var) = ival;
break;
default:
goto badtype;
}
}
} else if( item->type == kString ) {
ret = LoadDBvalue( f, date, key, *((string*)item->var) );
} else if( item->type == kTString ) {
ret = LoadDBvalue( f, date, key, *((TString*)item->var) );
} else if( item->type == kFloatV ) {
ret = LoadDBarray( f, date, key, *((vector<float>*)item->var) );
} else if( item->type == kDoubleV ) {
ret = LoadDBarray( f, date, key, *((vector<double>*)item->var) );
} else if( item->type == kIntV ) {
ret = LoadDBarray( f, date, key, *((vector<Int_t>*)item->var) );
} else if( item->type == kFloatM ) {
ret = LoadDBmatrix( f, date, key,
*((vector<vector<float> >*)item->var),
item->nelem );
} else if( item->type == kDoubleM ) {
ret = LoadDBmatrix( f, date, key,
*((vector<vector<double> >*)item->var),
item->nelem );
} else if( item->type == kIntM ) {
ret = LoadDBmatrix( f, date, key,
*((vector<vector<Int_t> >*)item->var),
item->nelem );
} else {
badtype:
const char* type_name;
if( item->type >= kDouble && item->type <= kObject2P )
type_name = var_type_name[item->type];
else
type_name = Form("(#%d)", item->type );
::Error( ::Here(here,loaddb_prefix.c_str()),
"Key \"%s\": Reading of data type \"%s\" not implemented",
type_name, key );
ret = -2;
break;
}
if( ret > 0 ) {
if( (search != 0 || item->search != 0) && *prefix ) {
DBRequest* newreq = new DBRequest[2];
if( newreq ) {
memcpy( newreq, item, sizeof(DBRequest) );
memset( newreq+1, 0, sizeof(DBRequest) );
newreq->search = 0;
Int_t newsearch = (item->search != 0) ? item->search : search;
string newprefix(prefix);
Int_t newlevel = ChopPrefix(newprefix);
Int_t ret2 = 1;
if( newsearch > 0 && newlevel >= newsearch )
ret2 = LoadDB( f, date, newreq, newprefix.c_str(), newsearch );
else if( newsearch < 0 )
ret2 = LoadDB( f, date, newreq, newprefix.c_str(), newsearch+1 );
delete [] newreq;
ret = ret2;
if( ret != 0 )
break;
goto nextitem;
} else {
ret = -5;
goto unexpected_error;
}
}
if( item->optional )
ret = 0;
else {
if( item->descript ) {
::Error( ::Here(here,loaddb_prefix.c_str()),
"Required key \"%s\" (%s) missing in the database.",
key, item->descript );
} else {
::Error( ::Here(here,loaddb_prefix.c_str()),
"Required key \"%s\" missing in the database.", key );
}
ret = 1+(item-req);
break;
}
} else if( ret == -128 ) {
::Error( ::Here(here,loaddb_prefix.c_str()),
"Text line too long. Fix the database!\n\"%s...\"",
errtxt.c_str() );
break;
} else if( ret == -129 ) {
::Error( ::Here(here,loaddb_prefix.c_str()),
"Number of matrix elements not evenly divisible by requested "
"number of columns. Fix the database!\n\"%s...\"",
errtxt.c_str() );
break;
} else if( ret < 0 ) {
unexpected_error:
::Error( ::Here(here,loaddb_prefix.c_str()),
"Program error when trying to read database key \"%s\". "
"CALL EXPERT!", key );
break;
}
}
nextitem:
item++;
}
if( --loaddb_depth == 0 )
loaddb_prefix.erase();
return ret;
}
Int_t THaAnalysisObject::LoadDB( FILE* f, const TDatime& date,
const TagDef* keys, const char* prefix,
Int_t search )
{
if( !keys )
return -1;
const TagDef* item = keys;
Int_t nreq = 0;
while( (item++)->name )
nreq++;
if( nreq == 0 )
return 0;
DBRequest *req = new struct DBRequest[nreq+1], *theReq = req;
if( !req )
return -3;
item = keys;
while( item->name ) {
theReq->name = item->name;
theReq->var = item->var;
theReq->type = item->type;
theReq->nelem = item->expected;
theReq->optional = item->fatal ? kFALSE : kTRUE;
theReq->search = 0;
theReq->descript = item->name;
item++; theReq++;
}
theReq->name = 0;
Int_t ret = LoadDB( f, date, req, prefix, search );
delete [] req;
return ret;
}
Int_t THaAnalysisObject::SeekDBconfig( FILE* file, const char* tag,
const char* label,
Bool_t end_on_tag )
{
if( !file || !tag || !*tag ) return 0;
string _label("[");
if( label && *label ) {
_label.append(label); _label.append("=");
}
ssiz_t llen = _label.size();
bool found = false, quit = false;;
const int LEN = 256;
char buf[LEN];
errno = 0;
off_t pos = ftello(file);
if( pos != -1 ) {
while( !errno && !found && !quit && fgets( buf, LEN, file)) {
size_t len = strlen(buf);
if( len<2 || buf[0] == '#' ) continue;
if( buf[len-1] == '\n') buf[len-1] = 0;
char* cbuf = ::Compress(buf);
string line(cbuf); delete [] cbuf;
ssiz_t lbrk = line.find(_label);
if( lbrk != string::npos && lbrk < line.size()-llen ) {
ssiz_t rbrk = line.find(']',lbrk+llen);
if( rbrk == string::npos ) continue;
if( line.substr(lbrk+llen,rbrk-lbrk-llen) == tag ) {
found = true;
break;
}
} else if( end_on_tag && IsTag(buf) )
quit = true;
}
}
if( errno ) {
perror( "THaAnalysisObject::SeekDBconfig" );
found = false;
}
if( !found && pos >= 0 )
fseeko( file, pos, SEEK_SET );
return found;
}
Int_t THaAnalysisObject::SeekDBdate( FILE* file, const TDatime& date,
Bool_t end_on_tag )
{
static const char* const here = "THaAnalysisObject::SeekDBdateTag";
if( !file ) return 0;
const int LEN = 256;
char buf[LEN];
TDatime tagdate(950101,0), prevdate(950101,0);
const bool kNoWarn = false;
errno = 0;
off_t pos = ftello(file);
if( pos == -1 ) {
if( errno )
perror(here);
return 0;
}
off_t foundpos = -1;
bool found = false, quit = false;
while( !errno && !quit && fgets( buf, LEN, file)) {
size_t len = strlen(buf);
if( len<2 || buf[0] == '#' ) continue;
if( buf[len-1] == '\n') buf[len-1] = 0;
string line(buf);
if( IsDBdate( line, tagdate, kNoWarn )
&& tagdate<=date && tagdate>=prevdate ) {
prevdate = tagdate;
foundpos = ftello(file);
found = true;
} else if( end_on_tag && IsTag(buf))
quit = true;
}
if( errno ) {
perror(here);
found = false;
}
fseeko( file, (found ? foundpos: pos), SEEK_SET );
return found;
}
vector<string> THaAnalysisObject::vsplit(const string& s)
{
vector<string> ret;
typedef string::size_type ssiz_t;
ssiz_t i = 0;
while ( i != s.size()) {
while (i != s.size() && isspace(s[i])) ++i;
ssiz_t j = i;
while (j != s.size() && !isspace(s[j])) ++j;
if (i != j) {
ret.push_back(s.substr(i, j-i));
i = j;
}
}
return ret;
}
ClassImp(THaAnalysisObject)
Last change: Sat Nov 7 21:26:41 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.