/* * Project: MoleCuilder * Description: creates and alters molecular systems * Copyright (C) 2010 University of Bonn. All rights reserved. * Please see the LICENSE file or "Copyright notice" in builder.cpp for details. */ /* * MatrixContainer.cpp * * Created on: Sep 15, 2011 * Author: heber */ // include config.h #ifdef HAVE_CONFIG_H #include #endif #include "CodePatterns/MemDebug.hpp" #include #include #include #include "CodePatterns/Log.hpp" #include "KeySetsContainer.hpp" #include "Fragmentation/helpers.hpp" #include "Helpers/defs.hpp" #include "Helpers/helpers.hpp" #include "MatrixContainer.hpp" /** Constructor of MatrixContainer class. */ MatrixContainer::MatrixContainer() { Header.resize(1); RowCounter.resize(1); ColumnCounter.resize(1); ColumnCounter[0] = -1; MatrixCounter = 0; }; /** Destructor of MatrixContainer class. */ MatrixContainer::~MatrixContainer() {} /** Either copies index matrix from another MatrixContainer or initialises with trivial mapping if NULL. * This either copies the index matrix or just maps 1 to 1, 2 to 2 and so on for all fragments. * \param *Matrix pointer to other MatrixContainer * \return true - copy/initialisation sucessful, false - dimension false for copying */ bool MatrixContainer::InitialiseIndices(class MatrixContainer *_container) { DoLog(0) && (Log() << Verbose(0) << "Initialising indices"); if (_container == NULL) { DoLog(0) && (Log() << Verbose(0) << " with trivial mapping." << endl); Indices.resize(MatrixCounter + 1); for(int i=MatrixCounter+1;i--;) { Indices[i].resize(RowCounter[i]); for(int j=RowCounter[i];j--;) Indices[i][j] = j; } } else { DoLog(0) && (Log() << Verbose(0) << " from other MatrixContainer." << endl); if (MatrixCounter != _container->MatrixCounter) return false; Indices.resize(MatrixCounter + 1); for(int i=MatrixCounter+1;i--;) { if (RowCounter[i] != _container->RowCounter[i]) return false; Indices[i].resize(_container->RowCounter[i]); for(int j=_container->RowCounter[i];j--;) { Indices[i][j] = _container->Indices[i][j]; //Log() << Verbose(0) << Indices[i][j] << "\t"; } //Log() << Verbose(0) << endl; } } return true; }; /** Parsing a number of matrices. * -# 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 * \param &input input stream * \param skiplines number of inital lines to skip * \param skiplines number of inital columns to skip * \param MatrixNr index number in Matrix array to parse into * \return parsing successful */ bool MatrixContainer::ParseMatrix(std::istream &input, int skiplines, int skipcolumns, size_t MatrixNr) { stringstream line; string token; char filename[1023]; if (input.fail()) { DoeLog(1) && (eLog()<< Verbose(1) << endl << "MatrixContainer::ParseMatrix: Unable to parse istream." << endl); //performCriticalExit(); return false; } // parse header if (Header.size() <= MatrixNr) Header.resize(MatrixNr); Header[MatrixNr] = std::string(""); char dummy[1024]; for (int m=skiplines+1;m--;) input.getline(dummy, 1023); line.str(dummy); for(int k=skipcolumns;k--;) line >> Header[MatrixNr]; Log() << Verbose(0) << line.str() << endl; // scan header for number of columns if (ColumnCounter.size() <= MatrixNr) ColumnCounter.resize(MatrixNr); ColumnCounter[MatrixNr]=0; while ( getline(line,token, '\t') ) { if (token.length() > 0) ColumnCounter[MatrixNr]++; } Log() << Verbose(0) << line.str() << endl; Log() << Verbose(1) << "ColumnCounter[" << MatrixNr << "]: " << ColumnCounter[MatrixNr] << "." << endl; if (ColumnCounter[MatrixNr] == 0) { DoeLog(0) && (eLog()<< Verbose(0) << "ColumnCounter[" << MatrixNr << "]: " << ColumnCounter[MatrixNr] << " from ostream." << endl); performCriticalExit(); } // scan rest for number of rows/lines if (RowCounter.size() <= MatrixNr) RowCounter.resize(MatrixNr); RowCounter[MatrixNr]=-1; // counts one line too much while (!input.eof()) { input.getline(filename, 1023); Log() << Verbose(0) << "Comparing: " << strncmp(filename,"MeanForce",9) << endl; RowCounter[MatrixNr]++; // then line was not last MeanForce if (strncmp(filename,"MeanForce", 9) == 0) { break; } } Log() << Verbose(1) << "RowCounter[" << MatrixNr << "]: " << RowCounter[MatrixNr] << " from input stream." << endl; if (RowCounter[MatrixNr] == 0) { DoeLog(0) && (eLog()<< Verbose(0) << "RowCounter[" << MatrixNr << "]: " << RowCounter[MatrixNr] << " from input stream." << endl); performCriticalExit(); } // allocate matrix if it's not zero dimension in one direction if (Matrix.size() <= MatrixNr) Matrix.resize(MatrixNr+1); if ((Matrix[MatrixNr].size() <= (size_t)RowCounter[MatrixNr] + 1) && (RowCounter[MatrixNr] > -1)) { Matrix[MatrixNr].resize(RowCounter[MatrixNr] + 2); for(int j=0;j<=RowCounter[MatrixNr];j++) { if ((Matrix[MatrixNr][j].size() <= (size_t)ColumnCounter[MatrixNr]) && (ColumnCounter[MatrixNr] > -1)) Matrix[MatrixNr][j].resize(ColumnCounter[MatrixNr]+1); // clear for(int k=0;k<=ColumnCounter[MatrixNr];k++) Matrix[MatrixNr][j][k] = 0; } } else { ELOG(1, "Matrix nr. " << MatrixNr << " has column and row count of (" << ColumnCounter[MatrixNr] << "," << RowCounter[MatrixNr] << "), could not allocate nor parse!"); return false; } // parse in each entry for this matrix input.clear(); input.seekg(ios::beg); for (int m=skiplines+1;m--;) input.getline(dummy, 1023); // skip header line.str(dummy); Log() << Verbose(0) << "Header: " << line.str() << endl; for(int k=skipcolumns;k--;) // skip columns in header too line >> filename; Header[MatrixNr] = line.str(); for(int j=0;j> filename; for(int k=0;(k> Matrix[MatrixNr][j][k]; Log() << Verbose(1) << " " << std::setprecision(2) << Matrix[MatrixNr][j][k] << endl; } } return true; }; /** 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::ParseFragmentMatrix(const std::string name, const std::string prefix, std::string suffix, int skiplines, int skipcolumns) { char filename[1023]; ifstream input; char *FragmentNumber = NULL; stringstream file; string token; // 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.bad()) { DoLog(0) && (Log() << Verbose(0) << endl << "MatrixContainer::ParseFragmentMatrix: Unable to open " << file.str() << ", is the directory correct?" << endl); return false; } while (!input.eof()) { input.getline(filename, 1023); stringstream zeile(filename); MatrixCounter++; } input.close(); DoLog(0) && (Log() << Verbose(0) << "Determined " << MatrixCounter << " fragments." << endl); DoLog(0) && (Log() << Verbose(0) << "Parsing through each fragment and retrieving " << prefix << suffix << "." << endl); Header.clear(); Matrix.clear(); RowCounter.clear(); ColumnCounter.clear(); Header.resize(MatrixCounter + 1); // one more each for the total molecule Matrix.resize(MatrixCounter + 1); // one more each for the total molecule RowCounter.resize(MatrixCounter + 1); ColumnCounter.resize(MatrixCounter + 1); for(int i=0; i < MatrixCounter;i++) { // open matrix file FragmentNumber = FixedDigitNumber(MatrixCounter, i); file.str(" "); file << name << FRAGMENTPREFIX << FragmentNumber << prefix << suffix; std::ifstream input(file.str().c_str()); DoLog(0) &&( Log() << Verbose(0) << "Opening " << file.str() << " ... " << endl); if (!ParseMatrix(input, skiplines, skipcolumns, i)) { input.close(); return false; } input.close(); delete[](FragmentNumber); } return true; }; /** Allocates and resets the memory for a number \a MCounter of matrices. * \param **GivenHeader Header line for each matrix * \param MCounter number of matrices * \param *RCounter number of rows for each matrix * \param *CCounter number of columns for each matrix * \return Allocation successful */ bool MatrixContainer::AllocateMatrix(StringVector GivenHeader, int MCounter, IntVector RCounter, IntVector CCounter) { MatrixCounter = MCounter; Header.resize(MatrixCounter + 1); Matrix.resize(MatrixCounter + 1); // one more each for the total molecule RowCounter.resize(MatrixCounter + 1); ColumnCounter.resize(MatrixCounter + 1); for(int i=MatrixCounter+1;i--;) { Header[i] = GivenHeader[i]; RowCounter[i] = RCounter[i]; ColumnCounter[i] = CCounter[i]; if (Matrix[i].size() <= RowCounter[i] + 2) Matrix[i].resize(RowCounter[i] + 1); for(int j=0;j<=RowCounter[i];j++) if (Matrix[i][j].size() <= ColumnCounter[i]+1) Matrix[i][j].resize(ColumnCounter[i]); // allocation with 0 is guaranted by STL } 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[i];k--;) Matrix[i][j][k] = 0.; return true; }; /** Scans all elements of MatrixContainer::Matrix for greatest absolute value. * \return greatest value of MatrixContainer::Matrix */ double MatrixContainer::FindMaxValue() { double max = Matrix[0][0][0]; for(int i=MatrixCounter+1;i--;) for(int j=RowCounter[i]+1;j--;) for(int k=ColumnCounter[i];k--;) if (fabs(Matrix[i][j][k]) > max) max = fabs(Matrix[i][j][k]); if (fabs(max) < MYEPSILON) max += MYEPSILON; return max; }; /** Scans all elements of MatrixContainer::Matrix for smallest absolute value. * \return smallest value of MatrixContainer::Matrix */ double MatrixContainer::FindMinValue() { double min = Matrix[0][0][0]; for(int i=MatrixCounter+1;i--;) for(int j=RowCounter[i]+1;j--;) for(int k=ColumnCounter[i];k--;) if (fabs(Matrix[i][j][k]) < min) min = fabs(Matrix[i][j][k]); if (fabs(min) < MYEPSILON) min += MYEPSILON; return min; }; /** 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[ KeySets.OrderSet[Order][CurrentFragment] ]) { DoeLog(0) && (eLog()<< Verbose(0) << "In fragment No. " << KeySets.OrderSet[Order][CurrentFragment] << " current force index " << m << " is greater than " << RowCounter[ KeySets.OrderSet[Order][CurrentFragment] ] << "!" << endl); performCriticalExit(); return false; } if (Order == SubOrder) { // equal order is always copy from Energies for(int l=ColumnCounter[ KeySets.OrderSet[SubOrder][j] ];l--;) // then adds/subtract each column Matrix[ KeySets.OrderSet[Order][CurrentFragment] ][m][l] += MatrixValues.Matrix[ KeySets.OrderSet[SubOrder][j] ][k][l]; } else { for(int l=ColumnCounter[ KeySets.OrderSet[SubOrder][j] ];l--;) Matrix[ KeySets.OrderSet[Order][CurrentFragment] ][m][l] -= Matrix[ KeySets.OrderSet[SubOrder][j] ][k][l]; } } //if ((ColumnCounter[ KeySets.OrderSet[SubOrder][j] ]>1) && (RowCounter[0]-1 >= 1)) //Log() << Verbose(0) << "Fragments[ KeySets.OrderSet[" << Order << "][" << CurrentFragment << "]=" << KeySets.OrderSet[Order][CurrentFragment] << " ][" << RowCounter[0]-1 << "][" << 1 << "] = " << Matrix[ KeySets.OrderSet[Order][CurrentFragment] ][RowCounter[0]-1][1] << endl; } } else { //Log() << Verbose(0) << "Fragment " << KeySets.OrderSet[SubOrder][j] << " is not contained in fragment " << KeySets.OrderSet[Order][CurrentFragment] << "." << endl; } } } //Log() << Verbose(0) << "Final Fragments[ KeySets.OrderSet[" << Order << "][" << CurrentFragment << "]=" << KeySets.OrderSet[Order][CurrentFragment] << " ][" << KeySets.AtomCounter[0]-1 << "][" << 1 << "] = " << Matrix[ KeySets.OrderSet[Order][CurrentFragment] ][KeySets.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 std::string name, const std::string prefix) { ofstream output; char *FragmentNumber = NULL; DoLog(0) && (Log() << Verbose(0) << "Writing fragment files." << endl); for(int i=0;i