![]() |
Bimetric 3+1 toolkit for spherical symmetry
|
00001 /** 00002 * @file gridOutput.h 00003 * @brief Writes checkpoints from the grid to the output data file. 00004 * @author Mikica Kocic 00005 * @copyright GNU General Public License (GPLv3). 00006 */ 00007 00008 #include <string> 00009 #include <vector> 00010 #include <deque> 00011 #include <cstdio> 00012 00013 /** Writes checkpoints from the grid to the output data file. 00014 */ 00015 class GridOutputWriter : GridUser 00016 { 00017 bool binaryFormat; //!< Write binary data to output file. 00018 FILE* outf; //!< Output stream for the results. 00019 std::string fileName; //!< Output filename 00020 std::string prefixFileName; //!< Filename that is copied out (reversed) before start 00021 std::vector<GF_Descriptor> output; //!< List of grid functions to write out 00022 00023 /** skipInfo declares a time point at which to set a new `mSkip` and `delta_t` for 00024 * the integration. 00025 */ 00026 struct skipInfo { Real t; Int mSkip; Real delta_t; }; 00027 00028 std::deque<skipInfo> skipSetter; //!< Apply new "skips" at the given time point 00029 00030 Int mSkip; //!< Output every `mSkip` rows 00031 Int nOut; //!< Number of grid points to get in the output 00032 Int nSkip; //!< Output every `nSkip` points 00033 Int chunkSize; //!< Size (in bytes) of the output chunk 00034 00035 Int checkNaNs_nFrom; //!< Check for NaNs from this `n` 00036 Int checkNaNs_nTo; //!< Check for NaNs to this `n` 00037 00038 /** Writes the results for a particular grid row to output. 00039 */ 00040 void write( Int m, Int nFrom, Int nTo, Int nSkip = 1 ) 00041 { 00042 for( Int n = nFrom; n < nTo; n += nSkip ) { 00043 GridPoint( *gridDriver, m, n ).write( outf, binaryFormat, output ); 00044 } 00045 } 00046 00047 public: 00048 00049 Int get_mSkip () const { return mSkip; } 00050 Int get_nOut () const { return nOut; } 00051 Int get_nSkip () const { return nSkip; } 00052 00053 Int get_nFrom () const { return checkNaNs_nFrom; } 00054 Int get_nTo () const { return checkNaNs_nTo; } 00055 00056 /** Retrieve the size (in bytes) of the output record. 00057 */ 00058 Int recordSizeInBytes () const { return chunkSize; } 00059 00060 /** Output size (number of grid functions). 00061 */ 00062 Int GFCount () const { return output.size(); } 00063 00064 /** Constructs the output writer as specified in the parameter file. 00065 */ 00066 GridOutputWriter( Parameters& params, UniformGrid& ug ) 00067 : GridUser( ug ), binaryFormat(false), outf(NULL) 00068 { 00069 chunkSize = 0; // Still not defined 00070 00071 output.reserve( 256 ); 00072 output.push_back( { fld::t, "t", "t" } ); 00073 output.push_back( { fld::r, "r", "r" } ); 00074 00075 params.get( "output.file", fileName, "stdout" ); 00076 params.get( "output.binary", binaryFormat, true ); 00077 params.get( "output.prefix", prefixFileName, "" ); 00078 00079 slog << "Output Writer:" << std::endl << std::endl 00080 << " file = " << fileName << ", binary = " << binaryFormat 00081 << std::endl; 00082 00083 params.get( "output.nOut", nOut, 10 ); 00084 params.get( "output.nSkip", nSkip, 1 ); 00085 params.get( "output.mSkip", mSkip, 1 ); 00086 00087 slog << " nOut = " << nOut << ", nSkip = " << nSkip << ", mSkip = " << mSkip 00088 << std::endl; 00089 00090 params.get( "checkNaNs.nFrom", checkNaNs_nFrom, nOut*9/10 ); 00091 params.get( "checkNaNs.nTo", checkNaNs_nTo, nOut ); 00092 00093 if( checkNaNs_nTo >= 0 ) { 00094 slog << " Check for NaNs: nFrom = " << checkNaNs_nFrom 00095 << ", nTo = " << checkNaNs_nTo << std::endl; 00096 } 00097 00098 slog << std::endl; 00099 00100 // Get skip setters 00101 // 00102 for( int i = 1; i < 10; ++i ) 00103 { 00104 std::string tag = "at" + std::to_string( i ); 00105 skipInfo info; 00106 params.get( tag + ".t", info.t, NAN ); 00107 params.get( tag + ".mSkip", info.mSkip, 1 ); 00108 params.get( tag + ".delta_t", info.delta_t, NAN ); 00109 if ( ! std::isnan( info.t ) ) { 00110 skipSetter.push_back( info ); 00111 } 00112 } 00113 00114 if( skipSetter.size() > 0 ) 00115 { 00116 slog << "Skip Setter:" << std::endl << std::endl; 00117 for( auto i: skipSetter ) { 00118 slog << " t = " << i.t << ", mSkip = " << i.mSkip 00119 << ", delta_t = " << i.delta_t << std::endl; 00120 } 00121 slog << std::endl; 00122 } 00123 } 00124 00125 ~GridOutputWriter () 00126 { 00127 close (); 00128 } 00129 00130 /** The given grid functions will be written to the output. 00131 */ 00132 void gridFunctions( const std::vector<GF_Descriptor>& gfs ) 00133 { 00134 output.insert( output.end(), gfs.begin(), gfs.end() ); 00135 // slog << " Added " << gfs.size () << " GFs to the output" << std::endl; 00136 } 00137 00138 /** Opens output data file for storing the results. 00139 */ 00140 bool open () 00141 { 00142 // Each record has two ghost regions in addition to the middle nOut/nSkip part 00143 // 00144 chunkSize = ( 2 * nGhost + nOut / nSkip ) * output.size() * sizeof(Real); 00145 00146 if( fileName == "stdout" ) { 00147 binaryFormat = false; 00148 outf = stdout; 00149 return false; 00150 } 00151 00152 // When using MPI, insert our rank number in the filename 00153 // 00154 if ( mpiSize() > 1 ) 00155 { 00156 size_t lastdot = fileName.find_last_of( "." ); 00157 fileName = fileName.substr( 0, lastdot ) + "-" 00158 + std::to_string( 1000 + mpiRank() ).substr( 1 ) 00159 + fileName.substr( lastdot ); 00160 } 00161 00162 outf = mpiRank() >= 4 ? NULL 00163 : fopen( fileName.c_str(), binaryFormat ? "wb+" : "w+" ); 00164 /* 00165 if( binaryFormat && prefixFileName != "" ) 00166 { 00167 std::ifstream pref( prefixFileName, std::ios::in | std::ios::binary ); 00168 if( pref ) 00169 { 00170 slog << "*** Appending the reversed-t: " << prefixFileName << std::endl; 00171 00172 pref.seekg( 0, std::ios::end ); // Go to end of file 00173 auto len = pref.tellg (); // Get the length of file (in bytes) 00174 00175 if ( ( len % chunkSize ) != 0 ) { 00176 std::cerr << "Error: Invalid grid alignment." << std::endl; 00177 return false; 00178 } 00179 00180 // While len > 0 means skip the last record (at t_0) 00181 for( len -= chunkSize; len > 0; len -= chunkSize ) 00182 { 00183 pref.seekg( len, std::ios::beg ); 00184 pref.read( (char*)&grid[0][0], chunkSize ); 00185 fwrite( &grid[0][0], chunkSize, 1, outf ); 00186 } 00187 pref.close (); 00188 } 00189 } 00190 */ 00191 return true; 00192 } 00193 00194 /** Closes the output file. 00195 */ 00196 void close () 00197 { 00198 if( outf != NULL ) { 00199 fclose( outf ); 00200 outf = NULL; 00201 } 00202 } 00203 00204 /** Writes the header to output file with the information about grid functions. 00205 */ 00206 void writeHeader () 00207 { 00208 if( outf != NULL ) 00209 { 00210 for( auto f: output ) { 00211 fprintf( outf, "%s -> \"%s\"\n", f.name.c_str(), f.tex.c_str() ); 00212 } 00213 fprintf( outf, "*** DATA ***\n"); 00214 } 00215 } 00216 00217 void write( Int m, Real cur_t, Real& delta_t ) 00218 { 00219 if ( outf != NULL ) 00220 { 00221 write( m, 0, nGhost ); 00222 write( m, nGhost, nGhost + std::min( nLen, nOut ), nSkip ); 00223 write( m, nGhost + nLen, nTotal ); 00224 } 00225 00226 if( skipSetter.size() > 0 && cur_t >= skipSetter.front().t ) 00227 { 00228 mSkip = skipSetter.front().mSkip; 00229 Real new_dt = skipSetter.front().delta_t; 00230 if( ! std::isnan( new_dt ) ) 00231 { 00232 delta_t = new_dt; 00233 slog << std::endl << std::endl << "*** at t = " << cur_t 00234 << " new mSkip = " << mSkip << ", delta_t = " << delta_t 00235 << std::endl << std::endl; 00236 } 00237 else 00238 { 00239 slog << std::endl << std::endl << "*** at t = " << cur_t 00240 << " new mSkip = " << mSkip << std::endl << std::endl; 00241 } 00242 skipSetter.pop_front (); 00243 } 00244 } 00245 };