Page MenuHomeHEPForge

TableCollection.cpp
No OneTemporary

TableCollection.cpp

//==============================================================================
// TableCollection.cpp
//
// Copyright (C) 2010-2013 Tobias Toll and Thomas Ullrich
//
// This file is part of Sartre version: 1.1
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation.
// This program is distributed in the hope that it will be useful,
// but without any warranty; without even the implied warranty of
// merchantability or fitness for a particular purpose. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
// Author: Thomas Ullrich
// Last update:
// $Date: 2014-03-27 20:52:41 +0000 (Thu, 27 Mar 2014) $
// $Author: thomas.ullrich@bnl.gov $
//==============================================================================
//
// Note that we do not take the lambda_A table into account when calculating
// the range since there is a fall back solution to calculate lambda if the
// table is not present. See class CrossSection.
//
//==============================================================================
#include "TSystemDirectory.h"
#include "TSystem.h"
#include "TList.h"
#include "EventGeneratorSettings.h"
#include "TableCollection.h"
#include "Table.h"
#include <string>
#include <sstream>
#include <cstdlib>
#include <limits>
#include <cmath>
#define PR(x) cout << #x << " = " << (x) << endl;
TableCollection::TableCollection() {/* no op */}
TableCollection::TableCollection(int A, DipoleModelType typ, int vmID)
{
init(A, typ, vmID);
}
TableCollection& TableCollection::operator=(const TableCollection& tc)
{
if (this != &tc) {
for (unsigned int i=0; i<mTables.size(); i++) // delete old
delete mTables[i];
mTables.clear(); // clear vector
for (unsigned int i=0; i<tc.mTables.size(); i++) // deep copy
mTables.push_back(new Table(*tc.mTables[i]));
}
return *this;
}
TableCollection::TableCollection(const TableCollection& tc)
{
for (unsigned int i=0; i<tc.mTables.size(); i++) // deep copy
mTables.push_back(new Table(*tc.mTables[i]));
}
TableCollection::~TableCollection()
{
for (unsigned int i=0; i<mTables.size(); i++)
delete mTables[i];
}
bool TableCollection::init(int A, DipoleModelType type, int vmID)
{
string saveCWD = gSystem->WorkingDirectory();
//
// Build directory path
//
stringstream pathstream;
pathstream << getenv("SARTRE_DIR") << "/tables/" << A << '/';
if (type == bSat)
pathstream << "bSat";
else if (type == bNonSat)
pathstream << "bNonSat";
else
pathstream << "bCGC";
pathstream << '/' << vmID;
string path = pathstream.str();
//
// Query list of all files in directory
// and create tables for each file ending
// in ".root", ignore others.
//
TSystemDirectory directory;
directory.SetDirectory(path.c_str());
TList *list = directory.GetListOfFiles();
if (!list) {
cout << "TableCollection::init(): Error, cannot find directory '" << path.c_str() << "' holding tables." << endl;
return false;
}
TIter next(list);
unsigned int numberOfTablesRead = 0;
while (TSystemFile* file = dynamic_cast<TSystemFile*>(next()) ) {
if (file->IsDirectory()) continue; // ignore directories
string name(file->GetName());
size_t pos = name.find(".root");
if (pos == string::npos || name.substr(pos) != string(".root")) continue; // ignore files not ending in .root
string fullpath = path + '/' + name;
Table *table = new Table;
if (table->read(fullpath.c_str())) {
mTables.push_back(table);
if (EventGeneratorSettings::instance()->verboseLevel() > 1)
cout << "Loaded table from file '" << fullpath.c_str() << "'." << endl;
}
numberOfTablesRead++;
}
//
// Cleanup
//
// Change ROOT directory back to directory we were before reading the tables.
// Otherwise reading tables interferes with user application.
//
list->Delete();
gSystem->ChangeDirectory(saveCWD.c_str());
if (!numberOfTablesRead) {
cout << "TableCollection::init(): Error, could not find any tables at '" << path.c_str() << "'." << endl;
return false;
}
return true;
}
bool TableCollection::tableExists(GammaPolarization pol, AmplitudeMoment mom) const
{
Table* currentTable;
for (unsigned int i=0; i<mTables.size(); i++) {
currentTable = mTables[i];
if (currentTable->polarization() != pol) continue;
if (currentTable->moment() != mom) continue;
return true;
}
return false;
}
bool TableCollection::available(double Q2, double W2, double t, GammaPolarization pol, AmplitudeMoment mom) const
{
//
// Check if table can provide this value
//
unsigned short nTables = 0;
Table* currentTable;
for (unsigned int i=0; i<mTables.size(); i++) {
currentTable = mTables[i];
if (currentTable->polarization() != pol) continue;
if (currentTable->moment() != mom) continue;
if (t >= currentTable->minT() && t <= currentTable->maxT()) {
if (Q2 >= currentTable->minQ2() && Q2 <= currentTable->maxQ2()) {
if (W2 >= currentTable->minW2() && W2 <= currentTable->maxW2()) {
nTables++;
}
}
}
}
if (nTables)
return true;
else
return false;
}
double TableCollection::get(double Q2, double W2, double t,
GammaPolarization pol, AmplitudeMoment mom) const
{
Table *table;
return get(Q2, W2, t, pol, mom, table);
}
double TableCollection::get(double Q2, double W2, double t,
GammaPolarization pol, AmplitudeMoment mom, Table *&table) const
{
//
// First get the tables that contain the necessary info.
// Later this should be a bit refined, here we simply
// loop over all tables to collect the relevant one(s).
//
vector<Table*> associatedTables;
Table* currentTable;
for (unsigned int i=0; i<mTables.size(); i++) {
currentTable = mTables[i];
if (currentTable->polarization() != pol) continue;
if (currentTable->moment() != mom) continue;
if (t >= currentTable->minT() && t <= currentTable->maxT()) {
if (Q2 >= currentTable->minQ2() && Q2 <= currentTable->maxQ2()) {
if (W2 >= currentTable->minW2() && W2 <= currentTable->maxW2()) {
associatedTables.push_back(currentTable);
}
}
}
}
if (associatedTables.size() == 0) {
table = 0;
if (mom != lambda_A) { // no warnings needed for lambda_A (can be calculated w/o tables)
cout << "TableCollection::get(): Warning, could not find any table containing t=" << t
<< ", Q2=" << Q2 << ", W2=" << W2 << endl;
cout << " Tables searched were for moment = " << (mom == mean_A ? "mean_A" : "mean_A2")
<< ", polarization = " << (pol == transverse ? 'T' : 'L') << endl;
}
return 0;
}
//
// In case of overlap of tables the following
// policy applies:
// 1. Use the table with the highest priority.
// 2. If there's more than one high priority table
// we average their values (if > 0).
//
unsigned int maxPriority = 0;
for (unsigned int i=0; i<associatedTables.size(); i++) {
if (associatedTables[i]->priority() > maxPriority)
maxPriority = associatedTables[i]->priority();
}
double result = 0;
int validCounter = 0;
table = 0;
for (unsigned int i=0; i<associatedTables.size(); i++) {
if (associatedTables[i]->priority() == maxPriority) {
double value = associatedTables[i]->get(Q2, W2, t);
if (value > 0) {
validCounter++;
result += value;
table = associatedTables[i];
}
}
}
if (validCounter) result /= validCounter;
return result;
}
void TableCollection::list(ostream& os, bool opt) const
{
for (unsigned int i=0; i<mTables.size(); i++)
mTables[i]->list(os, opt);
}
double TableCollection::minQ2() const
{
return minimumValue(0);
}
double TableCollection::maxQ2() const
{
return maximumValue(0);
}
double TableCollection::minW2() const
{
return minimumValue(1);
}
double TableCollection::maxW2() const
{
return maximumValue(1);
}
double TableCollection::minW() const {return sqrt(minW2());}
double TableCollection::maxW() const {return sqrt(maxW2());}
double TableCollection::minT() const
{
return minimumValue(2);
}
double TableCollection::maxT() const
{
return maximumValue(2);
}
double TableCollection::minimumValue(unsigned int kind) const // kind: Q2=0, W2=1, T=2;
{
double minPerTableType[4]; // L, L2, T, T2
fill(minPerTableType, minPerTableType+4, numeric_limits<float>::max());
for (unsigned int i=0; i<mTables.size(); i++) {
double val;
switch (kind) {
case (0):
val = mTables[i]->minQ2();
break;
case (1):
val = mTables[i]->minW2();
break;
default:
val = mTables[i]->minT();
break;
}
if (mTables[i]->isLongitudinal()) { // L or L2
if (mTables[i]->isMeanA())
minPerTableType[0] = min(minPerTableType[0],val); // L
else
minPerTableType[1] = min(minPerTableType[1],val); // L2
}
else { // T or T2
if (mTables[i]->isMeanA())
minPerTableType[2] = min(minPerTableType[2],val); // T
else
minPerTableType[3] = min(minPerTableType[3],val); // T2
}
}
double largestMin = *max_element(minPerTableType, minPerTableType+4);
return largestMin;
}
double TableCollection::maximumValue(unsigned int kind) const
{
double maxPerTableType[4]; // L, L2, T, T2
fill(maxPerTableType, maxPerTableType+4, -numeric_limits<float>::max());
for (unsigned int i=0; i<mTables.size(); i++) {
double val;
switch (kind) {
case (0):
val = mTables[i]->maxQ2();
break;
case (1):
val = mTables[i]->maxW2();
break;
default:
val = mTables[i]->maxT();
break;
}
if (mTables[i]->isLongitudinal()) { // L or L2
if (mTables[i]->isMeanA())
maxPerTableType[0] = max(maxPerTableType[0],val); // L
else
maxPerTableType[1] = max(maxPerTableType[1],val); // L2
}
else { // T or T2
if (mTables[i]->isMeanA())
maxPerTableType[2] = max(maxPerTableType[2],val); // T
else
maxPerTableType[3] = max(maxPerTableType[3],val); // T2
}
}
double smallestMax = *min_element(maxPerTableType, maxPerTableType+4);
return smallestMax;
}

File Metadata

Mime Type
text/x-c
Expires
Mon, Jan 20, 8:21 PM (12 h, 2 m)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
4242208
Default Alt Text
TableCollection.cpp (12 KB)

Event Timeline