/** \file parsing.cpp * * Declarations of various class functions for the parsing of value files. * */ // ======================================= INCLUDES ========================================== #include "helpers.hpp" #include "parser.hpp" // include config.h #ifdef HAVE_CONFIG_H #include #endif // ======================================= FUNCTIONS ========================================== /** Test if given filename can be opened. * \param filename name of file * \param test true - no error message, false - print error * \return given file exists */ bool FilePresent(const char *filename, bool test) { ifstream input; input.open(filename, ios::in); if (input == NULL) { if (!test) cout << endl << "Unable to open " << filename << ", is the directory correct?" << endl; return false; } input.close(); return true; }; /** Test the given parameters. * \param argc argument count * \param **argv arguments array * \return given inputdir is valid */ bool TestParams(int argc, char **argv) { ifstream input; stringstream line; line << argv[1] << FRAGMENTPREFIX << KEYSETFILE; return FilePresent(line.str().c_str(), false); }; // ======================================= CLASS MatrixContainer ============================= /** Constructor of MatrixContainer class. */ MatrixContainer::MatrixContainer() { Matrix = NULL; Indices = NULL; Header = NULL; MatrixCounter = 0; RowCounter = NULL; ColumnCounter = 0; }; /** Destructor of MatrixContainer class. */ MatrixContainer::~MatrixContainer() { if (Matrix != NULL) { for(int i=MatrixCounter;i--;) { if (RowCounter != NULL) { for(int j=RowCounter[i]+1;j--;) Free((void **)&Matrix[i][j], "MatrixContainer::~MatrixContainer: *Matrix[][]"); Free((void **)&Matrix[i], "MatrixContainer::~MatrixContainer: **Matrix[]"); } } if ((RowCounter != NULL) && (Matrix[MatrixCounter] != NULL)) for(int j=RowCounter[MatrixCounter]+1;j--;) Free((void **)&Matrix[MatrixCounter][j], "MatrixContainer::~MatrixContainer: *Matrix[MatrixCounter][]"); if (MatrixCounter != 0) Free((void **)&Matrix[MatrixCounter], "MatrixContainer::~MatrixContainer: **Matrix[MatrixCounter]"); Free((void **)&Matrix, "MatrixContainer::~MatrixContainer: ***Matrix"); } if (Indices != NULL) for(int i=MatrixCounter+1;i--;) { Free((void **)&Indices[i], "MatrixContainer::~MatrixContainer: *Indices[]"); } Free((void **)&Indices, "MatrixContainer::~MatrixContainer: **Indices"); Free((void **)&Header, "MatrixContainer::~MatrixContainer: *Header"); Free((void **)&RowCounter, "MatrixContainer::~MatrixContainer: *RowCounter"); }; /** Parsing a number of matrices. * -# First, count the number of matrices by counting lines in KEYSETFILE * -# Then, * -# construct the fragment number * -# open the matrix file * -# skip some lines (\a skiplines) * -# scan header lines for number of columns * -# scan lines for number of rows * -# allocate matrix * -# loop over found column and row counts and parse in each entry * -# Finally, allocate one additional matrix (\a MatrixCounter) containing combined or temporary values * \param *name directory with files * \param *prefix prefix of each matrix file * \param *suffix suffix of each matrix file * \param skiplines number of inital lines to skip * \param skiplines number of inital columns to skip * \return parsing successful */ bool MatrixContainer::ParseMatrix(char *name, char *prefix, string suffix, int skiplines, int skipcolumns) { char filename[1023]; ifstream input; char *FragmentNumber = NULL; stringstream file; Header = (char *) Malloc(sizeof(char)*1023, "MatrixContainer::ParseMatrix: *EnergyHeader"); // count the number of matrices MatrixCounter = -1; // we count one too much file << name << FRAGMENTPREFIX << KEYSETFILE; input.open(file.str().c_str(), ios::in); if (input == NULL) { cout << endl << "Unable to open " << file.str() << ", is the directory correct?" << endl; return false; } while (!input.eof()) { input.getline(filename, 1023); MatrixCounter++; } input.close(); cout << "Determined " << MatrixCounter << " fragments." << endl; cout << "Parsing through each fragment and retrieving " << prefix << suffix << "." << endl; Matrix = (double ***) Malloc(sizeof(double **)*(MatrixCounter+1), "MatrixContainer::ParseMatrix: ***Matrix"); // one more each for the total molecule RowCounter = (int *) Malloc(sizeof(int)*(MatrixCounter+1), "MatrixContainer::ParseMatrix: *RowCounter"); for(int i=MatrixCounter+1;i--;) { Matrix[i] = NULL; RowCounter[i] = -1; } for(int i=MatrixCounter+1;i--;) { // open matrix file stringstream line; FragmentNumber = FixedDigitNumber(MatrixCounter, i); file.str(" "); if (i != MatrixCounter) file << name << FRAGMENTPREFIX << FragmentNumber << "/" << prefix << suffix; else file << name << prefix << suffix; Free((void **)&FragmentNumber, "MatrixContainer::ParseMatrix: *FragmentNumber"); input.open(file.str().c_str(), ios::in); //cout << "Opening " << file.str() << " ... " << input << endl; if (input == NULL) { cerr << endl << "Unable to open " << file.str() << ", is the directory correct?" << endl; return false; } // skip some initial lines for (int m=skiplines+1;m--;) input.getline(Header, 1023); // scan header for number of columns line.str(Header); for(int k=skipcolumns;k--;) line >> Header; //cout << line.str() << endl; ColumnCounter=0; while (!line.eof()) { strcpy(filename,"@"); line >> filename; if (filename[0] != '@') ColumnCounter++; } //cout << line.str() << endl; //cout << "ColumnCounter: " << ColumnCounter << endl; // scan rest for number of rows/lines RowCounter[i]=-1; // counts one line too much while (!input.eof()) { input.getline(filename, 1023); //cout << "Comparing: " << strncmp(filename,"MeanForce",9) << endl; RowCounter[i]++; // then line was not last MeanForce if (strncmp(filename,"MeanForce", 9) == 0) { break; } } //cout << "RowCounter[" << i << "]: " << RowCounter[i] << " from file " << file.str() << "." << endl; // allocate matrix if it's not zero dimension in one direction if ((ColumnCounter > 0) && (RowCounter[i] > -1)) { Matrix[i] = (double **) Malloc(sizeof(double *)*(RowCounter[i]+1), "MatrixContainer::ParseMatrix: **Matrix[]"); // parse in each entry for this matrix input.clear(); input.seekg(ios::beg); for (int m=skiplines+1;m--;) input.getline(Header, 1023); // skip header line.str(Header); for(int k=skipcolumns;k--;) // skip columns in header too line >> filename; strncpy(Header, line.str().c_str(), 1023); for(int j=0;j> filename; for(int k=0;(k> Matrix[i][j][k]; //cout << " " << setprecision(2) << Matrix[i][j][k];; } //cout << endl; Matrix[i][ RowCounter[i] ] = (double *) Malloc(sizeof(double)*ColumnCounter, "MatrixContainer::ParseMatrix: *Matrix[RowCounter[i]][]"); for(int j=ColumnCounter;j--;) Matrix[i][ RowCounter[i] ][j] = 0.; } } else { cerr << "ERROR: Matrix nr. " << i << " has column and row count of (" << ColumnCounter << "," << RowCounter[i] << "), could not allocate nor parse!" << endl; } input.close(); } return true; }; /** Allocates and resets the memory for a number \a MCounter of matrices. * \param *GivenHeader Header line * \param MCounter number of matrices * \param *RCounter number of rows for each matrix * \param CCounter number of columns for all matrices * \return Allocation successful */ bool MatrixContainer::AllocateMatrix(char *GivenHeader, int MCounter, int *RCounter, int CCounter) { Header = (char *) Malloc(sizeof(char)*1024, "MatrixContainer::ParseMatrix: *EnergyHeader"); strncpy(Header, GivenHeader, 1023); MatrixCounter = MCounter; ColumnCounter = CCounter; Matrix = (double ***) Malloc(sizeof(double **)*(MatrixCounter+1), "MatrixContainer::ParseMatrix: ***Matrix"); // one more each for the total molecule RowCounter = (int *) Malloc(sizeof(int)*(MatrixCounter+1), "MatrixContainer::ParseMatrix: *RowCounter"); for(int i=MatrixCounter+1;i--;) { RowCounter[i] = RCounter[i]; Matrix[i] = (double **) Malloc(sizeof(double *)*(RowCounter[i]+1), "MatrixContainer::ParseMatrix: **Matrix[]"); for(int j=RowCounter[i]+1;j--;) { Matrix[i][j] = (double *) Malloc(sizeof(double)*ColumnCounter, "MatrixContainer::ParseMatrix: *Matrix[][]"); for(int k=ColumnCounter;k--;) Matrix[i][j][k] = 0.; } } return true; }; /** Resets all values in MatrixContainer::Matrix. * \return true if successful */ bool MatrixContainer::ResetMatrix() { for(int i=MatrixCounter+1;i--;) for(int j=RowCounter[i]+1;j--;) for(int k=ColumnCounter;k--;) Matrix[i][j][k] = 0.; return true; }; /** Sets all values in the last of MatrixContainer::Matrix to \a value. * \param value reset value * \param skipcolumns skip initial columns * \return true if successful */ bool MatrixContainer::SetLastMatrix(double value, int skipcolumns) { for(int j=RowCounter[MatrixCounter]+1;j--;) for(int k=skipcolumns;k RowCounter[ KeySet.OrderSet[Order][CurrentFragment] ]) { cerr << "In fragment No. " << KeySet.OrderSet[Order][CurrentFragment] << " current force index " << m << " is greater than " << RowCounter[ KeySet.OrderSet[Order][CurrentFragment] ] << "!" << endl; return false; } if (Order == SubOrder) { // equal order is always copy from Energies for(int l=ColumnCounter;l--;) // then adds/subtract each column Matrix[ KeySet.OrderSet[Order][CurrentFragment] ][m][l] += MatrixValues.Matrix[ KeySet.OrderSet[SubOrder][j] ][k][l]; } else { for(int l=ColumnCounter;l--;) Matrix[ KeySet.OrderSet[Order][CurrentFragment] ][m][l] -= Matrix[ KeySet.OrderSet[SubOrder][j] ][k][l]; } } //if ((ColumnCounter>1) && (RowCounter[0]-1 >= 1)) //cout << "Fragments[ KeySet.OrderSet[" << Order << "][" << CurrentFragment << "]=" << KeySet.OrderSet[Order][CurrentFragment] << " ][" << RowCounter[0]-1 << "][" << 1 << "] = " << Matrix[ KeySet.OrderSet[Order][CurrentFragment] ][RowCounter[0]-1][1] << endl; } } else { //cout << "Fragment " << KeySet.OrderSet[SubOrder][j] << " is not contained in fragment " << KeySet.OrderSet[Order][CurrentFragment] << "." << endl; } } } //cout << "Final Fragments[ KeySet.OrderSet[" << Order << "][" << CurrentFragment << "]=" << KeySet.OrderSet[Order][CurrentFragment] << " ][" << KeySet.AtomCounter[0]-1 << "][" << 1 << "] = " << Matrix[ KeySet.OrderSet[Order][CurrentFragment] ][KeySet.AtomCounter[0]-1][1] << endl; } return true; }; /** Writes the summed total fragment terms \f$F_{ij}\f$ to file. * \param *name inputdir * \param *prefix prefix before \a EnergySuffix * \return file was written */ bool MatrixContainer::WriteTotalFragments(const char *name, const char *prefix) { ofstream output; char *FragmentNumber = NULL; cout << "Writing fragment files." << endl; for(int i=0;i> Indices[i][j]; //cout << " " << Indices[i][j]; } //cout << endl; } Indices[MatrixCounter] = (int *) Malloc(sizeof(int)*RowCounter[MatrixCounter], "ForceMatrix::ParseIndices: *Indices[]"); for(int j=RowCounter[MatrixCounter];j--;) { Indices[MatrixCounter][j] = j; } input.close(); return true; }; /** Sums the forces and puts into last element of \a ForceMatrix::Matrix. * \param Matrix MatrixContainer with matrices (LevelCounter by ColumnCounter) with all the energies. * \param KeySet KeySetContainer with bond Order and association mapping of each fragment to an order * \param Order bond order * \return true if summing was successful */ bool ForceMatrix::SumSubForces(class ForceMatrix &Fragments, class KeySetsContainer &KeySet, int Order) { // sum forces for(int i=0;i RowCounter[MatrixCounter]) { cerr << "Current force index " << j << " is greater than " << RowCounter[MatrixCounter] << "!" << endl; return false; } if (j != -1) for(int k=2;k> KeySets[i][j]; cout << " " << KeySets[i][j]; } cout << endl; } input.close(); return true; }; /** Parse many body terms, associating each fragment to a certain bond order. * \return parsing succesful */ bool KeySetsContainer::ParseManyBodyTerms() { int Counter; cout << "Creating Fragment terms." << endl; // scan through all to determine maximum order Order=0; for(int i=FragmentCounter;i--;) { Counter=0; for(int j=AtomCounter[i];j--;) if (KeySets[i][j] != -1) Counter++; if (Counter > Order) Order = Counter; } cout << "Found Order is " << Order << "." << endl; // scan through all to determine fragments per order FragmentsPerOrder = (int *) Malloc(sizeof(int)*Order, "KeySetsContainer::ParseManyBodyTerms: *FragmentsPerOrder"); for(int i=Order;i--;) FragmentsPerOrder[i] = 0; for(int i=FragmentCounter;i--;) { Counter=0; for(int j=AtomCounter[i];j--;) if (KeySets[i][j] != -1) Counter++; FragmentsPerOrder[Counter-1]++; } for(int i=0;i FragmentCounter) || (SmallerSet > FragmentCounter)) // index out of bounds return false; for(int i=AtomCounter[SmallerSet];i--;) { intermediate = false; for (int j=AtomCounter[GreaterSet];j--;) intermediate = (intermediate || ((KeySets[SmallerSet][i] == KeySets[GreaterSet][j]) || (KeySets[SmallerSet][i] == -1))); result = result && intermediate; } return result; }; // ======================================= END =============================================