Bimetric 3+1 toolkit for spherical symmetry
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Defines
grid/gridOutput.h
Go to the documentation of this file.
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 };