#include <stdlib.h>
#include "TTableDescriptor.h"
#include "TTable.h"
#include "TClass.h"
#include "TDataMember.h"
#include "TDataType.h"
#include "Ttypes.h"
#include "TInterpreter.h"
TTableDescriptor *TTableDescriptor::fgColDescriptors = 0;
TString TTableDescriptor::fgCommentsName = ".comments";
TableClassImp(TTableDescriptor,tableDescriptor_st)
TTableDescriptor *TTableDescriptor::GetDescriptorPointer() const
{
return fgColDescriptors;
}
void TTableDescriptor::SetDescriptorPointer(TTableDescriptor *list)
{
fgColDescriptors = list;
}
void TTableDescriptor::SetCommentsSetName(const char *name)
{
fgCommentsName = name;
}
void TTableDescriptor::Streamer(TBuffer &R__b)
{
fSecondDescriptor = 0;
TTable::Streamer(R__b);
}
TTableDescriptor::TTableDescriptor(const TTable *parentTable)
: TTable("tableDescriptor",sizeof(tableDescriptor_st)), fRowClass(0),fSecondDescriptor(0)
{
if (parentTable) {
TClass *classPtr = parentTable->GetRowClass();
Init(classPtr);
}
else MakeZombie();
}
TTableDescriptor::TTableDescriptor(TClass *classPtr)
: TTable("tableDescriptor",sizeof(tableDescriptor_st)),fRowClass(0),fSecondDescriptor(0)
{
Init(classPtr);
}
TTableDescriptor::~TTableDescriptor()
{
#ifdef NORESTRICTIONS
if (!IsZombie()) {
for (Int_t i=0;i<GetNRows();i++) {
Char_t *name = (Char_t *)ColumnName(i);
if (name) delete [] name;
UInt_t *indxArray = (UInt_t *)IndexArray(i);
if (indxArray) delete [] indxArray;
}
}
#endif
if (fSecondDescriptor != this) {
delete fSecondDescriptor;
fSecondDescriptor = 0;
}
}
Int_t TTableDescriptor::AddAt(const void *c)
{
if (!c) return -1;
TDataSet *cmnt = MakeCommentField();
assert(cmnt!=0);
return TTable::AddAt(c);
}
void TTableDescriptor::AddAt(const void *c, Int_t i)
{
if (c) {
tableDescriptor_st *element = (tableDescriptor_st *)c;
const char *comment = element->fColumnName ? element->fColumnName : "N/A";
AddAt(*(tableDescriptor_st *)c,comment,i);
}
}
void TTableDescriptor::AddAt(TDataSet *dataset,Int_t idx)
{
TTable::AddAt(dataset,idx);
}
void TTableDescriptor::AddAt(const tableDescriptor_st &element,const char *commentText,Int_t indx)
{
TTable::AddAt(&element,indx);
TDataSet *cmnt = MakeCommentField();
assert(cmnt!=0);
TDataSet *comment = new TDataSet(element.fColumnName);
comment->SetTitle(commentText);
cmnt->AddAtAndExpand(comment,indx);
}
TString TTableDescriptor::CreateLeafList() const
{
const Char_t typeMapTBranch[]="\0FIISDiisbBC";
Int_t maxRows = NumberOfColumns();
TString string;
for (Int_t i=0;i<maxRows;i++){
if (i) string += ":";
UInt_t nDim = Dimensions(i);
UInt_t totalSize = 1;
UInt_t k = 0;
if (nDim) {
const UInt_t *indx = IndexArray(i);
if (!indx){
string = "";
Error("CreateLeafList()","Can not create leaflist for arrays");
return string;
}
for (k=0;k< nDim; k++) totalSize *= indx[k];
}
const Char_t *colName = ColumnName(i);
if (totalSize > 1) {
for ( k = 0; k < totalSize; k++) {
Char_t buf[10];
sprintf(buf,"_%d",k);
string += colName;
string += buf;
if (k==0) {
string += "/";
string += typeMapTBranch[ColumnType(i)];
}
if (k != totalSize -1) string += ":";
}
} else {
string += ColumnName(i);
string += "/";
string += typeMapTBranch[ColumnType(i)];
}
}
return string;
}
void TTableDescriptor::Init(TClass *classPtr)
{
fSecondDescriptor = 0;
SetType("tableDescriptor");
if (classPtr) {
fRowClass = classPtr;
SetName(classPtr->GetName());
LearnTable(classPtr);
}
else
MakeZombie();
}
void TTableDescriptor::LearnTable(const TTable *parentTable)
{
if (!parentTable) {
MakeZombie();
return;
}
LearnTable(parentTable->GetRowClass());
}
void TTableDescriptor::LearnTable(TClass *classPtr)
{
if (!classPtr) return;
if (!(classPtr->GetNdata())) return;
Char_t *varname;
tableDescriptor_st elementDescriptor;
ReAllocate(classPtr->GetListOfDataMembers()->GetSize());
Int_t columnIndex = 0;
TIter next(classPtr->GetListOfDataMembers());
TDataMember *member = 0;
while ( (member = (TDataMember *) next()) ) {
memset(&elementDescriptor,0,sizeof(tableDescriptor_st));
varname = (Char_t *) member->GetName();
#ifdef NORESTRICTIONS
elementDescriptor.fColumnName = StrDup(varname);
#else
elementDescriptor.fColumnName[0] = '\0';
strncat(elementDescriptor.fColumnName,varname,sizeof(elementDescriptor.fColumnName)-1);
#endif
if (member->IsaPointer() ) {
elementDescriptor.fTypeSize = sizeof(void *);
const char *typeName = member->GetTypeName();
elementDescriptor.fType = TTable::GetTypeId(typeName);
} else {
TDataType *memberType = member->GetDataType();
assert(memberType!=0);
elementDescriptor.fTypeSize = memberType->Size();
elementDescriptor.fType = TTable::GetTypeId(memberType->GetTypeName());
}
Int_t globalIndex = 1;
if (elementDescriptor.fType != kNAN) {
Int_t dim = 0;
if ( (dim = member->GetArrayDim()) ) {
elementDescriptor.fDimensions = dim;
#ifdef NORESTRICTIONS
elementDescriptor.fIndexArray = new UInt_t(dim);
#else
UInt_t maxDim = sizeof(elementDescriptor.fIndexArray)/sizeof(UInt_t *);
if (UInt_t(dim) > maxDim) {
Error("LearnTable","Too many dimenstions - %d", dim);
dim = maxDim;
}
#endif
for( Int_t indx=0; indx < dim; indx++ ){
elementDescriptor.fIndexArray[indx] = member->GetMaxIndex(indx);
globalIndex *= elementDescriptor.fIndexArray[indx];
}
}
}
else Error("LearnTable","Wrong data type for <%s> structure",classPtr->GetName());
elementDescriptor.fSize = globalIndex * (elementDescriptor.fTypeSize);
elementDescriptor.fOffset = member->GetOffset();
AddAt(elementDescriptor,member->GetTitle(),columnIndex); columnIndex++;
}
}
TTableDescriptor *TTableDescriptor::MakeDescriptor(const char *structName)
{
TTableDescriptor *dsc = 0;
TClass *cl = TClass::GetClass(structName, kTRUE);
assert(cl!=0);
dsc = new TTableDescriptor(cl);
return dsc;
}
TDataSet *TTableDescriptor::MakeCommentField(Bool_t createFlag){
TDataSet *comments = FindByName(fgCommentsName.Data());
if (!comments && createFlag)
comments = new TDataSet(fgCommentsName.Data(),this,kTRUE);
return comments;
}
Int_t TTableDescriptor::UpdateOffsets(const TTableDescriptor *newDescriptor)
{
Int_t maxColumns = NumberOfColumns();
Int_t mismathes = 0;
if ( (UInt_t(maxColumns) == newDescriptor->NumberOfColumns())
&& (memcmp(GetArray(),newDescriptor->GetArray(),sizeof(tableDescriptor_st)*GetNRows()) == 0)
) return mismathes;
for (Int_t colCounter=0; colCounter < maxColumns; colCounter++) {
Int_t colNewIndx = newDescriptor->ColumnByName(ColumnName(colCounter));
EColumnType newType = colNewIndx >=0 ? newDescriptor->ColumnType(colNewIndx): kNAN;
#ifdef __STAR__
if (newType == kInt) newType = kLong;
else if (newType == kUInt) newType = kULong;
#endif
if ( colNewIndx >=0
&& Dimensions(colCounter) == newDescriptor->Dimensions(colNewIndx)
&& ColumnType(colCounter) == newType) {
SetOffset(newDescriptor->Offset(colNewIndx),colCounter);
if (colNewIndx != colCounter) {
Printf("Schema evolution: \t%d column of the \"%s\" table has been moved to %d-th column\n",
colCounter,ColumnName(colCounter),colNewIndx);
mismathes++;
}
} else {
Printf("Schema evolution: \t%d column \"%s\" of %d type has been lost\n",
colCounter,ColumnName(colCounter),ColumnType(colCounter));
Printf(" Indx = %d, name = %s \n", colNewIndx, ColumnName(colCounter));
SetOffset(UInt_t(-1),colCounter);
mismathes++;
}
}
if (!mismathes && UInt_t(maxColumns) != newDescriptor->NumberOfColumns()) {
mismathes++;
Printf("Warning: One extra column has been introduced\n");
}
return mismathes;
}
Int_t TTableDescriptor::ColumnByName(const Char_t *columnName) const
{
const tableDescriptor_st *elementDescriptor = ((TTableDescriptor *)this)->GetTable();
Int_t i = -1;
if (!elementDescriptor) return i;
Int_t nRows = GetNRows();
char *bracket = 0;
if (nRows) {
char *name = StrDup(columnName);
if ((bracket = strchr(name,'[')) ) *bracket = 0;
for (i=0; i < nRows; i++,elementDescriptor++)
if (strcmp(name,elementDescriptor->fColumnName) == 0) break;
delete [] name;
}
if (i==nRows) i = -1;
if (bracket && !Dimensions(i)) {
i = -1;
Warning("ColumnByName","%s column contains a scalar value",columnName);
}
return i;
}
Int_t TTableDescriptor::Offset(const Char_t *columnName) const
{
Int_t indx = ColumnByName(columnName);
Int_t offset = -1;
if (indx >= 0 ) {
offset = Offset(indx);
const char *openBracket = 0;
if ( (openBracket = strchr(columnName,'[')) )
offset += atoi(openBracket+1)*TypeSize(indx);
}
return offset;
}
Int_t TTableDescriptor::ColumnSize(const Char_t *columnName) const
{
Int_t indx = ColumnByName(columnName);
if (indx >= 0 ) indx = ColumnSize(indx);
return indx;
}
Int_t TTableDescriptor::TypeSize(const Char_t *columnName) const
{
Int_t indx = ColumnByName(columnName);
if (indx >= 0 ) indx = TypeSize(indx);
return indx;
}
Int_t TTableDescriptor::Dimensions(const Char_t *columnName) const
{
Int_t indx = ColumnByName(columnName);
if (indx >= 0 ) indx = Dimensions(indx);
return indx;
}
TTable::EColumnType TTableDescriptor::ColumnType(const Char_t *columnName) const
{
Int_t indx = ColumnByName(columnName);
if (indx >= 0 ) indx = ColumnType(indx);
return EColumnType(indx);
}
Int_t TTableDescriptor::Sizeof() const
{
Int_t fullRowSize = 0;
if (RowClass() ) fullRowSize = RowClass()->Size();
else {
Int_t iLastRows = GetNRows()-1;
if (iLastRows >=0) fullRowSize = Offset(iLastRows) + ColumnSize(iLastRows);
}
return fullRowSize;
}
Last change: Wed Jun 25 08:54:05 2008
Last generated: 2008-06-25 08:54
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.